```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("玩游戏")
}
}