禁用Xcode 9的Swift 3 @objc推断
今天我安装了Xcode 9,并将我的项目转换为Swift 4.0。 令人惊讶的是,我的代码只需做很少的改动就可以兼容新的Swift。 然而,构建后出现一个警告:
The use of Swift 3 @objc inference in Swift 4 mode is deprecated. Please address deprecated @objc inference warnings, test your code with “Use of deprecated Swift 3 @objc inference” logging enabled, and disable Swift 3 @objc inference.
这里记录了我如何去掉这个警告:
- 在项目导航栏单击项目
- 选择目标(target)
- 点击Build Settings
- 在搜索框输入“inference”,快速找到设置
- 最后把“Swift 3 @objc Inference”设置为“Default”,重新构建项目
“Swift 3 @objc Inference”设置是做什么的?
在Swift 4之前,编译器对Objective-C自动提供了一些Swift声明。 例如,编译器会为NSObject子类的所有方法创建Objective-C入口点。 该机制称为@objc推断(@obj Inference)。
在Swift 4中,这种自动@objc推断已被废弃,因为生成所有这些Objective-C入口点有代价。 当“Swift 3 @objc Inference”设置为“On”时,它允许旧代码运行。 但是,它会提示这些需要解决的弃用警告,建议“修复”这些警告,并将设置切换到“Default”。新的Swift项目的默认为“Default”。
示例一:Swift 4之前
这段代码在Swift 4之前是有效的,因为方法foo自动暴露给Objective-C:
class MyClass: NSObject {
func foo() {}
func test() {
var cl: AnyObject
cl = MyClass()
cl.foo?() // Swift 4之前没有问题
}
}
示例二:Swift 4
试试在Swift 4中运行相同的代码:
class MyClass: NSObject {
func foo() {}
func test() {
var cl: AnyObject
cl = MyClass()
cl.foo?() // Error: Value of type 'AnyObject' has no member 'foo'
}
}
现在编译器显示一个错误,因为foo方法不再可以从Objective-C获得。 为方便起见,Swift 4代码迁移将“Swift 3 @objc Inference”设置为“On”,这使得我们的代码编译时出现以下警告:
Reference to instance method 'foo()' of 'MyClass' depends on '@objc' attribute inference deprecated in Swift 4.
让类成员可以在Objective-C里使用
如果我们想要从Objective-C访问声明,我们可以使用@objc注释标记个别成员:
class MyClass : NSObject {
@objc func foo() { } // 这个方法可以在Objective-C访问到
func bar() { } // 但这个方法不行
}
或者,如果我们希望类的所有成员都暴露于Objective-C,可以使用@objcMembers注释来标记该类:
@objcMembers // 所有的类成员都暴露于Objective-C
class MyClass : NSObject {
func foo() { }
func bar() { }
}
参考
- Limiting @objc inference Swift proposal.
- An answer on Stack Overflow.