如何从Swift中的Mirror子类中提取可选的类型值?

时间:2016-03-16 17:59:13

标签: swift reflection mirror

我有几个类似于以下

的快速类
public class Book {
  var title: String?
  var date: NSDate?
}

由于我需要访问属性的几个不同的类,我使用反射来运行类的属性:

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)
  }
}

但是我遇到的问题是Child对象没有被提取,因为它们是Type Any和内部可选类,因此不属于我的情况。如果我从String更改标题?对于String,它们确实有效,但我需要使用可选类型。

在上面的实现中,我可以将数据类型保留为String吗?和日期?并仍然从镜像中提取值?

1 个答案:

答案 0 :(得分:5)

似乎在Swift 2.x中这是不可能的。

由于属性是可选项,因此您必须分别转换为NSDate?String?,如下所示:

if let val = value as? NSDate? {
    // val is Optional<NSDate>
}

不幸的是,编译器不允许这样做(我不确定原因):// error: cannot downcast from 'protocol<>' to a more optional type 'NSDate?'

来自This answer

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)
        }
    }
}

但这取决于每个属性都有一个Mirror,而且似乎你无法遍历Mirror的孩子为他们检索Mirror