ExpressibleBy 字面量协议

128 阅读2分钟

自实际开发中我们会遇到这样的代码:

let aString = "this is a string"

swift是怎么知道这是一个String类型的初始化呢? 下面我们下介绍一下ExpressibleByStringLiteral,他只是ExpressibleBy家族中一位成员。

我们查看 String 这个类, Swift 实际上已经给 String 这个类实现了 ExpressibleByStringLiteral这个协议。所以可以通过这种方式来初始化一个String对象。通过查看文档: 这个协议已经被

  • CSLocalizedString
  • StaticString
  • NSMutableString
  • String
  • NSString
  • SubString
  • Selector 这几个东西实现了。 这个协议有什么用呢?其实我们可以通过实现这个协议,来帮助我们更简单的初始化一些东西。比如说: URL。 在实际开发中, 每次初始化一个 URL 都需要
let urls = URL.init(string: "https://www.baidu.com")

这样来初始化一个 URL 出来, 这个初始化出来的 URL 又是一个 optional 的,在使用的使用还需要给他解包什么的。但是每次都要这样实在是很烦。所以我们可以通过这个协议来简化这个过程。

extension URL: ExpressibleByStringLiteral {
    
public init(stringLiteral value: String) {
    guard let url = URL(string: "(value)") else {
        preconditionFailure("This url: (value) is not invalid")
    }
    self = url
}

这样就可以通过

let url: URL = "https://www.baidu.com"
let task = URLSession.shared.dataTask(with: url)
let task2 = URLSession.shared.dataTask(with: "https://www.baidu.com")

这种方式去初始化一个 URL 了,然后当这个 URL 不对的时候就会抛出异常。 当然还有一些其他的类, 也可以通过实现这个协议的方式来做这件事情。比如说: 我们有一个商品类

struct Product {
    var `id`: String
}

let p = Product(id: "xx")

extension Product: ExpressibleByStringLiteral {
    init(stringLiteral value: String) {
        self = Product(id: value)
    }
}
let p2: Product = "xx"

当然还可以直接通过一个字符串初始化一个 Controller 这种做法。

当然,本人还是不支持对这种自定义的类做这种事情,确实是一种非常风骚的做法。暂时没有想到十分具体的场景。但是知道这个小技巧还是很不错的。

不过,对URL Date 这类,在开发中经常会以字符串来表示,但是又是字符串东西来说,这样去写还是挺不错的。比如对 Date

extension Date: ExpressibleByStringLiteral {
    public init(stringLiteral value: String) {
        let dateformatter = DateFormatter()
        dateformatter.dateFormat = "YYYY-MM-dd"
        guard let date = dateformatter.date(from: value) else {
            preconditionFailure("This date: (value) is not invalid")
        }
        self = date
    }
}

let date: Date = "1990-09-09"

使用这种方式是不是可以做好多小轮子来使用呢。 其实基础类型基本都是通过字面量进行构造的:

let num: Int = 100
let flag: Bool = true
let str: String = "wyw"
let array: [String] = ["wyw", "liusan"]

而这些都有对应的字面量协议:

ExpressibleByNilLiteral // nil字面量协议
ExpressibleByIntegerLiteral // 整数字面量协议
ExpressibleByFloatLiteral // 浮点数字面量协议
ExpressibleByBooleanLiteral // 布尔值字面量协议
ExpressibleByStringLiteral // 字符串字面量协议
ExpressibleByArrayLiteral // 数组字面量协议