-
1.下边是一个简单的给类中属性赋值的例子,这样写是可以运行的。但是如果我想通过
setValue:forKey
对其进行赋值却报错Value of type 'KVCTestClass' has no member 'setValue'
.
class KVCTestClass {
var someValue: String = "你好"
}
let kvc = KVCTestClass()
print("赋值前边:\(kvc.someValue)")
kvc.someValue = "我很好"// 456
print("赋值后:\(kvc.someValue)")
问题分析
:首先我们都知道使用KVC的重要一点就是需要实现NSKeyValueCoding
的协议,因为所有的KVC核心的机制都在这个协议当中,而我们之所以在Objective-C 中可以直接使用不会报错,是因为我们在Objective-C 中创建类的时候xcode会默认为我们继承 NSObject基类。所以我们可以直接使用KVC进行读写操作。那么在swift中我们就需要手动去显示的继承一下NSObject类了。
-
2.更改后的代码如下
不是说继承
NSObject
这个类就可以了吗?,为啥下边的代码又报错了?error: Execution was interrupted, reason: signal SIGABRT. The process has been left at the point where it was interrupted, use "thread return -x" to return to the state before expression evaluation.
。
class KVCTestClass:NSObject{
var someValue: String = "你好"
}
let kvc = KVCTestClass()
print("赋值前边:\(kvc.someValue)")
kvc.setValue("我很好", forKey: "someValue")
print("赋值后:\(kvc.someValue)")
-
3.KVC最终正确的使用的版本
总结
:在swift中要使用KVC这种机制的话,继承NSObject这个类是必要条件,而且还需要显示的使用@objc
关键字来声明一下才可以正常使用。
class KVCTestClass:NSObject{
@objc var someValue: String = "你好"
}
let kvc = KVCTestClass()
print("赋值前边:\(kvc.someValue)")
kvc.setValue("我很好", forKey: "someValue")
print("赋值后:\(kvc.someValue)")
-
4.使用
keyPath
对属性进行读写操作。
class JCPerson : NSObject {
@objc var name:String = ""
init(dict:[String:Any]){
super.init()
setValuesForKeys(dict)
}
}
let person = JCPerson(dict: ["name":"码农晨仔"])
let name = person.value(forKeyPath: #keyPath(JCPerson.name))
print(name ?? "错误")
person.setValue("屌丝夜魅", forKeyPath: #keyPath(JCPerson.name))
print(person.name)