美文网首页Ios开发
Swift使用泛型进行快速的 UITableViewCell 复

Swift使用泛型进行快速的 UITableViewCell 复

作者: is0bnd | 来源:发表于2017-09-27 12:01 被阅读47次
    UITableView是我们经常使用的,通常我们的 cell 复用是按照以下的方法进行的
    class CustomCell: UITableViewCell {  
    }
    
    class ViewController: UITableViewController {
    
        let cellIdentifier = "let cellIdentifier"
        
        override func viewDidLoad() {
            super.viewDidLoad()
            tableView.register(CustomCell.self, forCellReuseIdentifier: cellIdentifier)
        }
    }
    
    extension ViewController {
        override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
            return 100
        }
        
        override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
            let cell = tableView.dequeueReusableCell(withIdentifier: cellIdentifier, for: indexPath) as! CustomCell
            return cell
        }
    }
    
    这样做有几个缺点
    1. 需要声明cellIdentifier, 保证不重复,且使用 xib/storyboard 时需要保证一致性,容易出错
    2. 返回的 cell 类型需要进行强制类型转换
    3. 多人协作命名一致性问题

    由于类名是不能重复的,为了解决这些问题,在最开始的时候,使用类名作为 cellIdentifier, 减少出现问题的可能,保证多人协作的一致性,像下面这样

    let cellIdentifier = "CustomCell"
    
    但是这并不方便,使用起来也基本没有太大的差别,再次尝试使用 UITableViewCell 类型直接进行注册,直接使用类型进行复用,方法也很简单,获取类名进行注册复用即可,方法如下
    // 给 UITableView 进行方法扩展,增加使用类型进行注册和复用的方法
    extension UITableView {
        func register(_ cellClass: UITableViewCell.Type) {
            let identifier = String(describing: cellClass)
            register(cellClass, forCellReuseIdentifier: identifier)
        }
        func dequeueReusableCell(with cellClass: UITableViewCell.Type, for indexPath: IndexPath) -> UITableViewCell {
            let identifier = String(describing: cellClass)
            return dequeueReusableCell(withIdentifier: identifier, for: indexPath)
        }
    }
    
    目前还剩最后一个问题,解决复用返回 cell 需要进行强制类型转换的问题, 这里用到了泛型来解决这个问题
    extension UITableView {
        func dequeueReusableCell<T: UITableViewCell>(with cellClass: T.Type, for indexPath: IndexPath) -> T {
            let identifier = String(describing: cellClass)
            return dequeueReusableCell(withIdentifier: identifier, for: indexPath) as! T
        }
    }
    
    这样就完成了整个的改造工作,用起来也非常简单
    class CustomCell: UITableViewCell {
        var index = 0
    }
    
    class ViewController: UITableViewController {
        
        override func viewDidLoad() {
            super.viewDidLoad()
            tableView.register(CustomCell.self)
        }
    }
    
    extension ViewController {
        override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
            return 100
        }
        
        override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
            let cell = tableView.dequeueReusableCell(with: CustomCell.self, for: indexPath)
            cell.index = indexPath.row
            return cell
        }
    }
    
    

    结语

    1. 存在问题:在 storyboard/xib 上使用需要复制类名作为 identifier,目前没找到好的解决办法
    2. 这里只举例了注册 UITableViewCell 复用的情况,其实头尾视图的注册复用, 以及 UICollectionView 相关的复用都可以使用同样的方法进行处理,自己动手尝试一下吧

    相关文章

      网友评论

        本文标题:Swift使用泛型进行快速的 UITableViewCell 复

        本文链接:https://www.haomeiwen.com/subject/twxuextx.html