如何在Swift中优雅地使用ReusableIdentifier

苹果为了保准UITableView视图的性能,使用了cell的重用机制,cell可以通过重用标示符(reusableIdentifier)进行复用,默认的注册cell和获取cell的方法中,需要传入一个字符串作重用标示符。但这种方式很容易出错,而且使用起来也相当别扭,一种普遍的解决方式,就是直接只用类名作为重用标示符:

1
2
3
tableview.registerClass(UITableViewCell.self, forCellReuseIdentifier: String(describing: UITableViewCell.self))

tableview.dequeueReusableCellWithIdentifier(String(describing: UITableViewCell.self))

但这种写法依然颇为繁琐,每次都要传入一个类,并把它转化成字符串。所幸,借助Swift的泛型特性,我们可以有更加优雅的实现方式。

使用协议

使用泛型来优化 TableView Cells 的使用体验这篇文章中,作者详细介绍了如何通过协议+泛型的方式,优化TableView Cells 的使用体验。具体的做法很简单,首先声明了一个协议,提供并默认实现了一个reuseIdentifier静态属性:

1
2
3
4
5
6
7
8
9
10

protocol Reusable {
static var reuseIdentifier: String { get }
}

extension Reusable {
static var reuseIdentifier: String {
return String(describing: self)
}
}

然后为UITableView提供一个注册和获取重用cell的方法:

1
2
3
4
5
6
7
8
9
extension UITableView {
func register<T: UITableViewCell>(_: T.Type) where T: Reusable {
self.register(T.self, forCellReuseIdentifier: T.reuseIdentifier)
}

func dequeueReusableCell<T: UITableViewCell>(indexPath: IndexPath) -> T where T: Reusable {
return self.dequeueReusableCell(withIdentifier: T.reuseIdentifier, for: indexPath) as! T
}
}

这样只要cell遵守了Reusable协议,就可以通过上面两个方法注册复用cell了。具体的代码和使用,请阅读原文:使用泛型来优化 TableView Cells 的使用体验

这种方式的确是比原生方法方便了不少,但依然有两个缺点:

  1. 采用这种方式的cell必须遵循Reusable协议,虽然我们可以通过让UITableViewCell遵守这个协议的方式,避免每个UITableViewCell子cell都写一遍,但依然还有不便之处。
  2. 这种方式产生的reusableIdentifier是固定的,就是类名,因此不适用于一个cell有不同的样式,需要不同的reusableIdentifier的情况。

实际上使用协议做一层封装完全没必要,直接通过扩展增加方法就可以了。

通过方法扩展

我们在注册、复用cell的时候,最根本的是需要有一个reusableIdentifier,上述方法是在一个协议中提供了产生reusableIdentifier的默认实现,因此需要遵循协议,而且无法对reusableIdentifier进行自定义。其实如果我们只是想要一个类名作为reusableIdentifier,根本不需要任何协议或结构体等多余的类型,只需要将上面我们为UITableView扩展的两个个方法中,ReuseIdentifier的获取方式改成String(describing: T.self)即可:

1
2
3
4
5
6
7
8
9
extension UITableView {
func register<T: UITableViewCell>(_: T.Type) {
self.register(T.self, forCellReuseIdentifier: String(describing: T.self))
}

func dequeueReusableCell<T: UITableViewCell>(indexPath: IndexPath) -> T {
return self.dequeueReusableCell(withIdentifier: String(describing: T.self), for: indexPath) as! T
}
}

但这种方式还是有第二种缺点,无法自定义reusableIdentifier,要做到这点,也很简单,在上述两个方法中加上一个参数即可,然后给参数提供一个默认值。

1
2
3
4
5
6
7
8
9
extension UITableView {
func register<T: UITableViewCell>(_: T.Type, reuseIdentifier: String = String(describing: T.self)) {
self.register(T.self, forCellReuseIdentifier: reuseIdentifier)
}

func dequeueReusableCell<T: UITableViewCell>(indexPath: IndexPath, reuseIdentifier: String = String(describing: T.self)) -> T {
return self.dequeueReusableCell(withIdentifier: reuseIdentifier, for: indexPath) as! T
}
}
坚持原创技术分享,您的支持将鼓励我继续创作!