在 swift 中搜索字典时如何使用可选链接?

How to use optional chaining while searching through a dictionary in swift?

我想使用 if let 安全地搜索 swift 字典中的值,并确保它是类型安全的,因为我越来越深入字典。该字典包含包含 NSArray 的字典,后者包含更多字典。

第一次尝试我的代码是这样的:

  if let kkbox = ticket["KKBOX"] as? Dictionary<String, AnyObject> {
        if let kkboxDlUrlDict = kkbox["kkbox_dl_url_list"] as? Dictionary<String, AnyObject> {
            if let kkboxDlUrlArray = kkboxDlUrlDict["kkbox_dl_url"] as? NSArray {
                for dict in kkboxDlUrlArray {
                    if let name = dict["name"] as? String {
                        if name == mediaType.rawValue {
                            urlStr = dict["url"] as String
                        }
                    }
                }
            } else { return nil }
        } else { return nil }
    } else { return nil }

如何将其缩短为一两行?

我意识到如果它是 2 层我可以链接它。这有效:

 if let kkboxDlUrlArray = ticket["KKBOX"]?["kkbox_dl_url_list"] as? NSArray {

    }

但是任何比这更长的链都不会编译。

有没有办法多次链接字典?

谢谢

您可以链接,但每一步都要适当向下转换:

if let kkboxDlUrlArray = ((((ticket["KKBOX"] as? Dictionary<String, AnyObject>)?["kkbox_dl_url_list"]) as? Dictionary<String, AnyObject>)?["kkbox_dl_url"]) as? NSArray {
    for dict in kkboxDlUrlArray {
        println(dict)
    }
}

虽然看起来不太好 - 它只有一行,但不可读。

就我个人而言,在不使用任何花哨的函数式方法的情况下,我会通过一个可选绑定使链条更加明确:

let kkbox = ticket["KKBOX"] as? Dictionary<String, AnyObject>
let kkboxDlUrlDict = kkbox?["kkbox_dl_url_list"] as? Dictionary<String, AnyObject>
if let kkboxDlUrlArray = kkboxDlUrlDict?["kkbox_dl_url"] as? NSArray {
    for dict in kkboxDlUrlArray {
        println(dict)
    }
}

在我看来更容易阅读,并且没有不必要的缩进

好像swift 1.2增加了这个功能。

"More powerful optional unwrapping with if let — The if let construct can now unwrap multiple optionals at once, as well as include intervening boolean conditions. This lets you express conditional control flow without unnecessary nesting."

https://developer.apple.com/swift/blog/

或者,您可以使用此扩展,它模仿了 NSDictionary 中曾经存在的原始 valueForKeyPath 方法,但由于某种原因被砍掉了:

Swift 4.1

extension Dictionary where Key: ExpressibleByStringLiteral {

func valueFor(keyPath: String) -> Any? {
    var keys = keyPath.components(separatedBy: ".")
    var val : Any = self
    while keys.count > 0 {
        if let key = keys[0] as? Key {
            keys.remove(at: 0)
            if let dic = val as? Dictionary<Key, Value> {
                if let leaf = dic[key] {
                    val = leaf
                } else {
                    return nil
                }
            } else {
                return nil
            }
        } else {
            return nil
        }
    }
    return val
}

您的代码将显示为:

if let array = ticket.valueFor("KKBOX.kkbox_dl_url_list.kkbox_dl_url") as? [] {
  // do something with array
}