如何在Swift中优雅地使用ReusableIdentifier重用标示符?

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友 微信微信
查看查看157 回复回复1 收藏收藏 分享淘帖 转播转播 分享分享 微信
查看: 157|回复: 1
收起左侧

如何在Swift中优雅地使用ReusableIdentifier重用标示符?

[复制链接]
swifter 发表于 2016-8-16 17:29:29 | 显示全部楼层 |阅读模式
快来登录
获取最新的苹果动态资讯
收藏热门的iOS等技术干货
拷贝下载Swift Demo源代码

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

[Swift] 纯文本查看 复制代码
tableview.registerClass(UITableViewCell.self, forCellReuseIdentifier: String(UITableViewCell.self))

tableview.dequeueReusableCellWithIdentifier(String(UITableViewCell.self))


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

如何在Swift中优雅地使用ReusableIdentifier重用标示符?

如何在Swift中优雅地使用ReusableIdentifier重用标示符? - 敏捷大拇指 - 如何在Swift中优雅地使用ReusableIdentifier重用标示符?





1、使用协议

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

[Swift] 纯文本查看 复制代码
protocol Reusable: class {
  static var reuseIdentifier: String { get }
}

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


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

[Swift] 纯文本查看 复制代码
func registerReusableCell<T: UITableViewCell where T: Reusable>(_: T.Type) {
     self.registerClass(T.self, forCellReuseIdentifier: T.reuseIdentifier)
  }

  func dequeueReusableCell<T: UITableViewCell where T: Reusable>(indexPath indexPath: NSIndexPath) -> T {
    return self.dequeueReusableCellWithIdentifier(T.reuseIdentifier, forIndexPath: indexPath) as! T
  }


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

这种方式的确是比原生的方法方便了不少,但还是有一个不太方便的地方,那就是cell必须遵守了Reusable协议,虽然我们可以通过让UITableViewCell遵守这个协议的方式,避免每个UITableViewCell子cell都写一遍,但这依然还不是最理想的解决方式。最理想的解决方式,应该是只需要调用方法,不需要UITableViewCell做任何修改,为此我们可以使用结构体,而非协议来实现。




2、使用结构体

我们可以使用泛型定义一个ReusableIdentifier结构体,有一个identifier的不变量:

[Swift] 纯文本查看 复制代码
public struct ReusableIdentifier <T: UIView> {
  let identifier: String

  init() {
    identifier = String(T.self)
  }
}


然后为UITableView实现一个register方法,这个方法只需要传入一个类型即可:

[Swift] 纯文本查看 复制代码
extension UITableView {
    func register<T: UITableViewCell>(_: T.Type) {
    registerClass(T.self, forCellReuseIdentifier: ReusableIdentifier<T>().identifier)
    }
  }


如此,注册的时候就非常简单:tableview.register(UITableViewCell.self)。

同样的,可以为UITableView实现一个dequeue方法:

[Swift] 纯文本查看 复制代码
@warn_unused_result
  func dequeue<T: UICollectionViewCell>(indexPath: NSIndexPath) -> T {
    let rid = ReusableIdentifier<T>()
    guard let cell = dequeueReusableCellWithReuseIdentifier(rid.identifier, forIndexPath: indexPath) as? T else {
      assertionFailure("No identifier(\(rid.identifier)) found for \(T.self)")
      return T.init()
    }
    return cell
  }


使用的时候只需要指定cell的类型,传入indexPath即可:

[Swift] 纯文本查看 复制代码
let cell: UITableViewCell = tableview.dequeue(indexPath)


通过引入一个结构体,利用泛型特性,不需要对已有的类型做任何修改,只需要替换注册和复用cell时调用的方法,我们就可以非常优雅的复用Tableview Cell。

参考上面的方法,我们可以借助ReusableIdentifier结构体,为UICollectionView实现相应的方法。




作者:小悟空

都看到这里了,就把这篇资料推荐给您的好朋友吧,让他们也感受一下。

回帖是一种美德,也是对楼主发帖的尊重和支持。

*声明:敏捷大拇指是全球最大的Swift开发者社区、苹果粉丝家园、智能移动门户,所载内容仅限于传递更多最新信息,并不意味赞同其观点或证实其描述;内容仅供参考,并非绝对正确的建议。本站不对上述信息的真实性、合法性、完整性做出保证;转载请注明来源并加上本站链接,敏捷大拇指将保留所有法律权益。如有疑问或建议,邮件至marketing@swifthumb.com

*联系:微信公众平台:“swifthumb” / 腾讯微博:@swifthumb / 新浪微博:@swifthumb / 官方QQ一群:343549891(满) / 官方QQ二群:245285613 ,需要报上用户名才会被同意进群,请先注册敏捷大拇指

嗯,不错!期待更多好内容,支持一把:
支持敏捷大拇指,用支付宝支付10.24元 支持敏捷大拇指,用微信支付10.24元

评分

参与人数 1金钱 +10 贡献 +10 专家分 +10 收起 理由
Anewczs + 10 + 10 + 10 32个赞!专家给力!

查看全部评分

小开开 发表于 2016-8-17 21:52:12 | 显示全部楼层
就不应该要什么identifier
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

做任务,领红包。
我要发帖

分享扩散

都看到这里了,就把这资料推荐给您的好朋友吧,让他们也感受一下。
您的每一位朋友访问此永久链接后,您都将获得相应的金钱积分奖励
热门推荐

合作伙伴

Swift小苹果

  • 北京治世天下科技有限公司
  • ©2014-2016 敏捷大拇指
  • 京ICP备14029482号
  • Powered by Discuz! X3.1 Licensed
  • swifthumb Wechat Code
  •   
快速回复 返回顶部 返回列表