Bootstrap

解读短小精悍的 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!"
}

首先,上面三段代码是等价的,但是代码逐渐简化了。

用图来表示执行结构就为:

参考