Se7*_*ays 6 reflection mirror swift
我有几个类似于以下类似的快速类
public class Book {
var title: String?
var date: NSDate?
}
Run Code Online (Sandbox Code Playgroud)
由于我需要访问属性的几个不同的类,我使用反射来运行类的属性:
let myBook = Book()
myBook.title = "Hello"
myBook.date = NSDate()
let mirror = Mirror(reflecting: myBook)
var propsArr = [(key: String?, value: Any)]()
let mirrorChildrenCollection = AnyRandomAccessCollection(mirror.children)!
if mirrorChildrenCollection.count > 0 {
propsArr += mirrorChildrenCollection
}
//iterate through properties
for case let (label?, value) in propsArr {
print (label, value)
if let val = value as? NSDate {
var extractedDate = val
print(extractedDate)
}
else if let val = value as? String {
var extractedTitle = val
print (extractedTitle)
}
}
Run Code Online (Sandbox Code Playgroud)
但我有一个问题,即Child对象没有被提取,因为它们是Type Any和内部可选类,因此不属于我的情况.如果我从String更改标题?对于String,它们确实有效,但我需要使用可选类型.
在上面的实现中我可以更改什么以将数据类型保留为String?和日期?并仍然从镜像中提取值?
看来这在Swift 2.x中是不可能的.
由于属性是可选项,因此您必须分别转换为NSDate?和String?:
if let val = value as? NSDate? {
// val is Optional<NSDate>
}
Run Code Online (Sandbox Code Playgroud)
不幸的是,编译器不会允许这种(我不知道为什么): // error: cannot downcast from 'protocol<>' to a more optional type 'NSDate?'.
来自bubuxu的这个答案提供了一个聪明的解决方法,如果您拥有Mirror每个属性,它将起作用.镜像的displayStyle属性告诉您它是否是可选的,然后您可以手动提取包装的值.这样可行:
let child = Mirror(reflecting: myBook.date)
child.displayStyle
if child.displayStyle == .Optional {
if child.children.count == 0 {
// .None
} else {
// .Some
let (_, some) = child.children.first!
if let val = some as? NSDate {
print(val)
}
}
}
Run Code Online (Sandbox Code Playgroud)
但这取决于Mirror每个属性都有一个,而且似乎你不能遍历一个Mirror孩子Mirror为他们检索s.