更智能安全的Key Value Coding

2020-03-20  本文已影响0人  醉看红尘这场梦

在Swift里,有一种间接访问类属性的方法,叫做#keyPath。来看下面的例子:

class Foo: NSObject {
    @objc var bar = "bar"
    @objc var baz = [1, 2, 3, 4]
}

var foo = Foo()
print(foo.bar)
foo.bar = "BAR"

除了像上面这样直接访问属性之外,我们还可以这样:

var bar = foo.value(forKeyPath: #keyPath(Foo.bar))
foo.setValue("BAR", forKeyPath: #keyPath(Foo.bar))

这就是Cocoa中的KVC机制,在Objective-C中它可以很好的工作,但移植到Swift之后,它的不足就显现出来了:

类型更安全的KeyPath

为此,Swift 4中设计了更智能的KeyPath,像这样:

let barKeyPath = \Foo.bar
var bar = foo[keyPath: barKeyPath]
foo[keyPath: barKeyPath] = "BAR"

其中,\Foo.bar就是Swift 4中新的key path用法,它是一个独立的类型,带有属性的类型信息,因此,编译器会发现错误类型的赋值,而不会把这个错误推迟到运行时:

// error: cannot assign value of type 'Int' to type 'String'
foo[keyPath: barKeyPath] = 1

适应性更好的KeyPath

除了类型安全之外,新的KeyPath用法适应性也更好,我们无须要求FooNSObject的派生类,也不必用@objc修饰属性,把Foo改成这样,之前的代码仍旧可以正常工作:

class Foo {
    var bar = "bar"
    var baz = [1, 2, 3, 4]
}

甚至,我们都无须要求Foo是一个classstruct属性同样可以使用新的KeyPath:

struct Foo {
    var bar = "bar"
    var baz = [1, 2, 3, 4]
}

可以在KeyPath中使用下标操作符(尚未实现)

SE-0161中提议的新型KeyPath当前还没有全部实现完成,但在它的设计里,KeyPath中是可以使用下标操作符的,因此,为了访问baz中的第1个元素,我们本可以这样:

// error: INTERNAL ERROR: feature not implemented
foo[keyPath: \Foo.baz[1]]

但由于,这个功能在录制这个视频的时候还没有实现,因此,我们会看到一个编译错误。这并不重要,我们只要了解这个用法就好了。

上一篇下一篇

猜你喜欢

热点阅读