综合技术

Swift3、4中的@objc、@objcMembers和dynamic

微信扫一扫,分享到朋友圈

Swift3、4中的@objc、@objcMembers和dynamic
0

Objective-C 对象是基于运行时的,方法或属性使用动态派发 ,在运行调用时再决定实际调用的具体实现。而 Swift 为了追求性能,如果没有特殊需要的话,是不会在运行时再来决定这些的。也就是说,Swift 类型的成员或者方法在编译时就已经决定,而运行时便不再需要经过一次查找,而可以直接使用。

Objective-C 中所有类都继承自 NSObject
,Swift 中的类如果要供 Objective-C 调用,必须也继承自 NSObject

@objc

  1. @objc
    修饰符的根本目的是用来暴露接口给 Objective-C 的运行时(类、协议、属性和方法等)

  2. 添加 @objc
    修饰符并不意味着这个方法或者属性会采用 Objective-C 的方式变成动态派发,Swift 依然可能会将其优化为静态调用

  3. @objc
    修饰符的隐式添加:

    • Swift 3 中继承自 NSObject
      的类,不需要手动添加 @objc
      ,编译器会给所有的非 private
      的类和成员加上 @objc
      private
      接口想要暴露给 Objective-C 需要 @objc
      的修饰

      button.addTarget(self, action: #selector(backButtonTapped), for: .touchUpInside)
      @objc private func backButtonTapped() { }
      func backButtonTapped() { }
      复制代码
    • Swift 4 中继承自 NSObject
      的类的隐式 @objc
      自动添加,只会发生在以下四种情况:

      • 重写了父类的 Objective-C 方法
      • 实现了一个 Objective-C 的协议
      • @IBAction
        @IBOutlet
        关键字的修饰
      • @NSManaged
        关键字的修饰
  4. 使用 @objc
    可以修改 Swift 接口暴露到 Objective-C 后的名字

    @objc(Squirrel)
    class Белка: NSObject {
        @objc(color)
        var цвет: Цвет = .Красный
    
        @objc(hideNuts:inTree:)
        func прячьОрехи(количество: Int, вДереве дерево: Дерево) { }
    }
    复制代码

@objcMembers

Swift4 后继承自 NSObject
的类不再隐式添加 @objc
关键字,但在某些情况下非常依赖 Objective-C 的运行时(如 XCTest),所以在 Swift4 中提供了 @objcMembers
关键字,对类和子类、扩展和子类扩展重新启用 @objc
推断。

@objcMembers
class MyClass : NSObject {
  func foo() { }             // implicitly @objc

  func bar() -> (Int, Int)   // not @objc, because tuple returns
      // aren't representable in Objective-C
}

extension MyClass {
  func baz() { }   // implicitly @objc
}

class MySubClass : MyClass {
  func wibble() { }   // implicitly @objc
}

extension MySubClass {
  func wobble() { }   // implicitly @objc
}
复制代码

使用 @objc
@nonobjc
可以指定开启或关闭某一 extension
中的所有方法的 @objc
推断。

class SwiftClass { }

@objc extension SwiftClass {
  func foo() { }            // implicitly @objc
  func bar() -> (Int, Int)  // error: tuple type (Int, Int) not
      // expressible in @objc. add @nonobjc or move this method to fix the issue
}

@objcMembers
class MyClass : NSObject {
  func wibble() { }    // implicitly @objc
}

@nonobjc extension MyClass {
  func wobble() { }    // not @objc, despite @objcMembers
}
复制代码

dynamic

当前 Swift 的动态性依赖于 Objective-C,Swift3 中 dynamic
就隐式包含了 @objc
的意思,但考虑到以后版本的 Swift 语言和运行时将会自支持 dynamic
而不再依赖于 Objective-C,所以在 Swift4 中将 dynamic
@objc
含义进行了抽离。

class MyClass {
  dynamic func foo() { }       // error: 'dynamic' method must be '@objc'
  @objc dynamic func bar() { } // okay
}
复制代码

参考:
Limiting @objc
inference

阅读原文...


稀土掘金

Android Handler 源码探索

上一篇

巨量引擎王牌IP再启动,今年营销中国行有何看点?

下一篇

您也可能喜欢

评论已经被关闭。

插入图片
Swift3、4中的@objc、@objcMembers和dynamic

长按储存图像,分享给朋友