使用可变参数调用 Objective-C 初始值设定项
Calling Objective-C initializer with variadic arguments
我正在尝试重新使用 Objective-C class,即 TSAlertView, into a Swift project. The problem is that the class uses an initializer with variadic arguments. I followed the same approach suggested at this Whosebug question,如果我使用 [=52],我的代码可以在 iOS 模拟器中运行=] Air 但如果我使用 iPad Retina 则不会。该代码也会在真实的 iPad 3.
上崩溃
我能够创建一个显示相同问题的玩具示例。
TestClass.h
#import <Foundation/Foundation.h>
@interface TestClass : NSObject
@property NSArray *titles;
- (id)initWithTitle:(NSString *)title otherButtonTitlesVA:(va_list)args;
@end
TestClass.m
#import "TestClass.h"
@implementation TestClass
- (id)initWithTitle:(NSString *)title otherButtonTitlesVA:(va_list)args {
NSMutableArray *titles = [NSMutableArray array];
if ((self = [super init])) {
[titles addObject:title];
id arg;
if ((arg = va_arg(args, id)) && [arg isKindOfClass:[NSString class]]) { // this causes an EXC_BAD_ACCESS on iPad Retina
[titles addObject:(NSString*)arg];
while ( nil != ( arg = va_arg( args, id ) ) )
{
if ( ![arg isKindOfClass: [NSString class] ] )
return nil;
[titles addObject:(NSString*)arg];
}
}
}
self.titles = [NSArray arrayWithArray:titles];
return self;
}
@end
测试类+Swift.swift
import Foundation
extension TestClass {
convenience init(title:String?, otherButtonTitles:CVarArgType...)
{
self.init(title: title, otherButtonTitlesVA:getVaList(otherButtonTitles))
}
}
ViewController.swift
import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
let testObject1 = TestClass(title: "First", otherButtonTitles: "Second", "Third")
let testObject2 = TestClass(title: "First") // this causes the initializer to crash on iPad Retina
NSLog("%@", testObject1.titles)
NSLog("%@", testObject2.titles)
}
}
尝试使用 EXC_BAD_ACCESS
创建 testObject2
时代码崩溃。我的代码有什么问题吗?为什么 iPad Air 显示出与 iPad Retina 不同的行为?
编辑:好的,我认为问题在于 Objective-C 代码需要一个 nil
终止列表。如何从 Swift 传递 nil
终止的 va_list
?
如您所知,Objective-C 代码需要 nil
终止列表。没有它,
的行为
if ((arg = va_arg(args, id)) && [arg isKindOfClass:[NSString class]]) { ... }
如果实际传递的参数列表已用尽,则未定义。
nil
是 Objective-C 中的 NULL 指针,您可以将其附加到您的便利初始值设定项中:
extension TestClass {
convenience init(title:String?, otherButtonTitles : CVarArgType...)
{
let nullPtr = UnsafePointer<Void>()
let otherTitles : [CVarArgType] = otherButtonTitles + [nullPtr]
self.init(title: title, otherButtonTitlesVA: getVaList(otherTitles))
}
}
这似乎在 32 位和 64 位平台上都能正常工作。
Swift3 的更新:
extension TestClass {
convenience init(title:String?, otherButtonTitles : CVarArg...)
{
let otherTitles : [CVarArg] = otherButtonTitles + [Int(0)]
self.init(title: title, otherButtonTitlesVA: getVaList(otherTitles))
}
}
如 https://bugs.swift.org/browse/SR-5046 中所述,使用 Int
大小的零是在
可变参数列表。
我正在尝试重新使用 Objective-C class,即 TSAlertView, into a Swift project. The problem is that the class uses an initializer with variadic arguments. I followed the same approach suggested at this Whosebug question,如果我使用 [=52],我的代码可以在 iOS 模拟器中运行=] Air 但如果我使用 iPad Retina 则不会。该代码也会在真实的 iPad 3.
上崩溃我能够创建一个显示相同问题的玩具示例。
TestClass.h
#import <Foundation/Foundation.h>
@interface TestClass : NSObject
@property NSArray *titles;
- (id)initWithTitle:(NSString *)title otherButtonTitlesVA:(va_list)args;
@end
TestClass.m
#import "TestClass.h"
@implementation TestClass
- (id)initWithTitle:(NSString *)title otherButtonTitlesVA:(va_list)args {
NSMutableArray *titles = [NSMutableArray array];
if ((self = [super init])) {
[titles addObject:title];
id arg;
if ((arg = va_arg(args, id)) && [arg isKindOfClass:[NSString class]]) { // this causes an EXC_BAD_ACCESS on iPad Retina
[titles addObject:(NSString*)arg];
while ( nil != ( arg = va_arg( args, id ) ) )
{
if ( ![arg isKindOfClass: [NSString class] ] )
return nil;
[titles addObject:(NSString*)arg];
}
}
}
self.titles = [NSArray arrayWithArray:titles];
return self;
}
@end
测试类+Swift.swift
import Foundation
extension TestClass {
convenience init(title:String?, otherButtonTitles:CVarArgType...)
{
self.init(title: title, otherButtonTitlesVA:getVaList(otherButtonTitles))
}
}
ViewController.swift
import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
let testObject1 = TestClass(title: "First", otherButtonTitles: "Second", "Third")
let testObject2 = TestClass(title: "First") // this causes the initializer to crash on iPad Retina
NSLog("%@", testObject1.titles)
NSLog("%@", testObject2.titles)
}
}
尝试使用 EXC_BAD_ACCESS
创建 testObject2
时代码崩溃。我的代码有什么问题吗?为什么 iPad Air 显示出与 iPad Retina 不同的行为?
编辑:好的,我认为问题在于 Objective-C 代码需要一个 nil
终止列表。如何从 Swift 传递 nil
终止的 va_list
?
如您所知,Objective-C 代码需要 nil
终止列表。没有它,
if ((arg = va_arg(args, id)) && [arg isKindOfClass:[NSString class]]) { ... }
如果实际传递的参数列表已用尽,则未定义。
nil
是 Objective-C 中的 NULL 指针,您可以将其附加到您的便利初始值设定项中:
extension TestClass {
convenience init(title:String?, otherButtonTitles : CVarArgType...)
{
let nullPtr = UnsafePointer<Void>()
let otherTitles : [CVarArgType] = otherButtonTitles + [nullPtr]
self.init(title: title, otherButtonTitlesVA: getVaList(otherTitles))
}
}
这似乎在 32 位和 64 位平台上都能正常工作。
Swift3 的更新:
extension TestClass {
convenience init(title:String?, otherButtonTitles : CVarArg...)
{
let otherTitles : [CVarArg] = otherButtonTitles + [Int(0)]
self.init(title: title, otherButtonTitlesVA: getVaList(otherTitles))
}
}
如 https://bugs.swift.org/browse/SR-5046 中所述,使用 Int
大小的零是在
可变参数列表。