Swift中的@objc、@objcMembers关键字和dynamic

939 阅读3分钟
```js
本文已参与新人创作礼活动,一起开启掘金创作之路。
# @objc
## 用它来干什么
在Swift中所有继承自NSObject的类,仍然保留了Objective-c的动态性,如果想要使用它的动态性就需要加上@objc关键字
## 哪些情况会用到

```js
1.Protocol如果是optional(非必须实现)的,必须加上@objc\
2.利用#selector调用的方法,被调用的方法须加上@objc\
3.使用kvc时\
4.NSPredicate筛选\
5.oc与swift混合开发,swift方法/属性需要被oc调用的,要加上@objc\
6.swift的枚举需要被oc使用的

1.Protocol如果是optional(非必须实现)的,必须加上@objc\

如下这样的一个协议,是两个方法都必须实现的

protocol PlayerDelegate: class {
    
    func playerReadyToPlay(player: CustomMediaPlayer, totalTime: Double)
    
    func playerCacheProgress(player: CustomMediaPlayer, progress: Float)
        
}

如果里面某些方法是非必须实现的,则需要在方法前加上@objc optional,在protocol前也需要加上@objc,如下:

@objc protocol PlayerDelegate: class {
    
   @objc optional func playerReadyToPlay(player: CustomMediaPlayer, totalTime: Double)
    
   @objc optional func playerCacheProgress(player: CustomMediaPlayer, progress: Float)
        
}

2.利用#selector调用的方法,被调用的方法须加上@objc\

这种情况用的也挺多 在Swift中#selector,相当于oc中的@selector,而@selector会在运行时进行方法调用。所以在Swift中使用#selector调用方法,需要在方法前加上@objc。拿button调用方法举例:

button.addTarget(self, action: #selector(click), for: .touchUpInside)

@objc private func click() {

}

3.使用kvc时\

kvc其实也是利用了oc的运行时特性,所以使用kvc也需要在属性前加上@objc,

class People: NSObject {
    @objc var name = ""
    var age = 10    
}

///使用valueforkey获取name值
let people = People()
people.name = "张三"
people.age = 15
let name = people.value(forKey: "name")
print(name)

people.setValue("王麻子", forKey: "name")

4.NSPredicate筛选\

NSPredicate是利用对象属性的key进行筛选的属性,要加上@objc,实际上NSPredicate也是利用了对象的oc对象的kvc特性

5.oc与swift混合开发,swift方法/属性需要被oc调用的,要加上@objc\

oc与swift混合开发,swift方法/属性需要被oc调用的,属性或方法前需要加上@objc。

class People: NSObject {
    @objc var name = ""
    
    @objc func playGame() {
        print("玩游戏")
    }
}

Swift类中的属性和方法加上@objc

People *people = [[People alloc] init];
people.name = @"111";
[people playGame];

注:类前不需要加@objc,只要是继承自NSObject的类,被oc调用时,系统会默认为被调用的类加为@objc。

6.swift的枚举需要被oc使用的

Swift的枚举被oc使用时,必须在enum前加上@objc,且Swift的枚举必须是Int类型,否则无法被oc使用

@objc enum PeopleSex: Int {
    case female = 0
    case male = 1
}

Swift的枚举,必须是Int类型,才能被oc调用

@objc与@objcMembers的区别

在Swift中,继承自NSObject的类如果有比较多的属性或方法都需要加上@objc的话,会多比较多的代码。那么可以利用@objcMembers减少代码。被@objcMembers修饰的类,会默认为类、子类、类扩展和子类扩展的所有属性和方法都加上@objc。当然如果想让某一个扩展关闭@objc,则可以用@nonobjc进行修饰。

**

@objcMembers
class People: NSObject {
    var name = ""
    var age = 10
    
    func playGame() {
        print("玩游戏")
    }
}