发布于 5年前

禁用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.

这里记录了我如何去掉这个警告:

  1. 在项目导航栏单击项目
  2. 选择目标(target)
  3. 点击Build Settings
  4. 在搜索框输入“inference”,快速找到设置
  5. 最后把“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() { }
}
©2020 edoou.com   京ICP备16001874号-3