昨天看了公众号SwiftGG翻译组
的文章《Swift:让人眼前一亮的初始化方式》,自己动手敲了一篇,只能说,手感真特么爽。
先看看比较普通的写法,最普通的就不说了,先看这种通过Block创建对象的写法,优点:代码整合在一个闭包里面,可读性强。
let _ = { () -> UILabel in
let label = UILabel()
view.addSubview(label)
label.text = "myLabel"
label.font = UIFont.systemFont(ofSize: 18)
label.center = self.view.center
label.bounds = CGRect.init(x: 0, y: 0, width: 100, height: 50)
return label
}()
但如果要在一个ViewDidLoad
创建多个Label呢,那多个闭包里面怎么给相应的Label命名呢?都叫label
还是不同的命名呢?这是强迫症和选择困难症并发的节奏。
接着看下面的写法,在Block外初始化一个对象,再传进闭包进行设值,统一用$0取参数对象,不用再去烦如何命名了。当然也可以命名,参考后文的写法
不过有个缺点:里面的$0没法自动联想出属性,手敲几行代码你就会蓝瘦香菇
let _: UILabel = {
view.addSubview($0)
$0.text = "测试"
$0.font = UIFont.systemFont(ofSize: 18)
$0.center = self.view.center
$0.bounds = CGRect.init(x: 0, y: 0, width: 100, height: 50)
return $0
}(UILabel())
大概理解上面的代码后,接着再看使用Then协议库的初始化写法,比上面的代码更简洁,不用写return了,而且里面的$0还能自动联想。优点:暂时没有吐槽点
// 1.0 带参数,可自行命名
let _ = UILabel().then_Any { (label) in
label.backgroundColor = .blue
label.font = UIFont.systemFont(ofSize: 18)
label.textAlignment = .center
label.text = "Then协议库写法_1.0"
label.frame = CGRect.init(x: 20, y: 200, width: 150, height: 40)
// 不会循环引用(已测试)
self.view.addSubview(label)
}
// 1.1 (推荐)无参数,无需命名,用$0取参数,可自动联想属性(推荐)
let _ = UILabel().then_Any {
$0.backgroundColor = .blue
$0.font = UIFont.systemFont(ofSize: 18)
$0.textAlignment = .center
$0.text = "Then库写法_1.1"
$0.frame = CGRect.init(x: 200, y: 260, width: 150, height: 40)
self.view.addSubview($0)
}
// 2.0 带参数,可自行命名
let label_AnyO = UILabel().then { (label) in
label.backgroundColor = .blue
label.font = UIFont.systemFont(ofSize: 18)
label.textAlignment = .center
label.text = "Then库写法_2.0"
label.frame = CGRect.init(x: 200, y: 260, width: 150, height: 40)
}
self.view.addSubview(label_AnyO)
// 2.1 (推荐)无参数,无需命名,用$0取参数,可自动联想属性
let _ = UILabel().then {
$0.backgroundColor = .blue
$0.font = UIFont.systemFont(ofSize: 18)
$0.textAlignment = .center
$0.text = "Then库写法_2.1"
$0.frame = CGRect.init(x: 200, y: 260, width: 150, height: 40)
self.view.addSubview($0)
}
/// 栗子:某工厂方法
class func label(withFrame frame: CGRect, text: String?, font: UIFont) -> UILabel {
return UILabel.init(frame: frame).then(block: { (label) in
label.text = text
label.textColor = UIColor.black
label.backgroundColor = UIColor.white
label.font = font
})
}
其实Then协议库只有十几行代码,但是能带来非常好的体验感,简约实用,不会出现循环引用。下面的代码可以复制到一个swift file中,直接用。
import Foundation
public protocol Then {}
extension Then where Self: Any {
public func then_Any( block: (inout Self) -> Void) -> Self {
var copy = self
block(©)
return copy
}
}
extension Then where Self: AnyObject {
public func then( block: (Self) -> Void) -> Self {
block(self)
return self
}
}
extension NSObject: Then {}
我所有Swift3.0练习Demo都放到了Github上,并且在不断更新。
Swift3.0朝圣之路-全集地址