解读短小精悍的 Then 框架
框架的核心代码不到 80 行,但是目前已经获得了 3.5k 的 star,着实让人佩服。
所以我感觉一个优秀的框架不在于多么庞大,而在是不是切实解决了开发者的某个痛点,提供了确切的帮助。
接下来,我们就来剖析一下这个短小精悍的框架。
方法的定义:
public protocol Then {}
extension Then where Self: AnyObject {
/// Makes it available to set properties with closures just after initializing.
///
/// let label = UILabel().then {
/// $0.textAlignment = .center
/// $0.textColor = UIColor.black
/// $0.text = "Hello, World!"
/// }
@inlinable
public func then(_ block: (Self) throws -> Void) rethrows -> Self {
try block(self)
return self
}
}
把上面的定义分成一个个的单词来看:
AnyObject
是一个空的协议,所有的类都实现了该协议。
可以放在 语句中,用于限定为类型。具体怎么限定,下面会提到。
self (小写的 s)
分成 3 种情况:
在方法内的 ,比如常见的 ,此处的 表示具体的实例(类实例,结构体实例,枚举实例,等等);
在某个 类型 后面的 ,比如 ,表示类型本身;
在某个 实例 后面也可以加 ,比如 ,还是表示这个实例本身(感觉没什么用)。
Self(大写的 S)
经常用在和协议相关的地方,指代的是实现该协议的类型本身,也包括了这个类型的子类类型。
上面代码中,先看协议扩展()中出现的 :
extension Then where Self: AnyObject {
}
其中:
用来指定限制条件。
,此处的 就表示实现 协议的类型本身,或者这个类型的子类的类型。 就表示当前类型需要符合(实现了) 协议。
连在一起就是,对实现了 协议,并且符合 协议的类型进行扩展。
再看 方法定义上出现的 :
// 为了方便理解,我把 throws,rethrows 移除了
func then(_ block: (Self) -> Void) -> Self {
}
其中:
方法名为 ;
接收一个名为 的参数,参数类型为: 的闭包,这个闭包为:接收一个类型为 的参数,返回值类型为 ;
返回值的类型为 ;
这里的 同样表示实现 协议的类型本身,或者这个类型的子类的类型。
通过上面的解释后,我们再来完整的看一下代码:
extension Then where Self: AnyObject {
@inlinable
public func then(_ block: (Self) throws -> Void) rethrows -> Self {
try block(self)
return self
}
}
即:对实现了 协议,并且符合 协议的类型进行扩展;在扩展中添加了一个 方法,该方法接受一个类型为 的闭包,返回值类型为该类型本身(即:)。在 方法内执行了外界传入的 ,并返回了当前实例(即:)。
Real World Example
拿一个真实的例子来举例:
let label = UILabel().then({ label in
label.textAlignment = .center
label.textColor = UIColor.black
label.text = "Hello, World!"
})
等价于
let label = UILabel().then { label in
label.textAlignment = .center
label.textColor = UIColor.black
label.text = "Hello, World!"
}
也等价于
let label = UILabel().then {
$0.textAlignment = .center
$0.textColor = UIColor.black
$0.text = "Hello, World!"
}
首先,上面三段代码是等价的,但是代码逐渐简化了。
用图来表示执行结构就为:
