zoukankan      html  css  js  c++  java
  • Swift TabeleViewCell dequeueReusableCellWithIdentifier 使用的新的细节,原来现在可以这样

    今天在看官方的TableView Guide,突然想起来最近写的一个代码中实现tableViewCell复用的时候有点问题:

    var cell = UITableViewCell(style: UITableViewCellStyle.Subtitle, reuseIdentifier: identifer)
    
    cell.textLabel.text = myPeripherals[indexPath.row].name

    这里可以看出来,每次调用tableView:cellForRowAtIndexPath: 时都会新建一个cell对象,这样肯定是不好的。

    在OC中的正统的实现方式应该是:

    先是用dequeueReusableCellWithIdentifier去将队列中已有的cell,则将其取出,重新利用。

    如果cell为nil,则重新实例化一个TableViewCell对象,再使用~~~~

    所以这样我就像将我有问题的Swift代码改回来,如下:

    但是这样会崩溃,不管是写成optional value的形式:

    var cell? = tableView.dequeueReusableCellWithIdentifier(identifer) as? UITableViewCell

    都会报同样的错误,搞了半天才发现,原来这个错误不是cell == nil导致的,因为“Create new cell”没有被输出过。

    出错的原因就在cell.detailTextLabel上出现,说明cell没有被设置成Subtitle的风格。

    再去看苹果官方的TableViewSuite Sample Code,中对tableViewCell的调用发现

    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    
        static NSString *MyIdentifier = @"MyIdentifier";
    
        /*
         Retrieve a cell with the given identifier from the table view.
         The cell is defined in the main storyboard: its identifier is MyIdentifier, and  its selection style is set to None.
         */
        UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:MyIdentifier];
    
        // Set up the cell.
        NSString *timeZoneName = [self.timeZoneNames objectAtIndex:indexPath.row];
        cell.textLabel.text = timeZoneName;
    
        return cell;
    }

    这里也是直接调用dequeueReusableCellWithIdentifier:MyIdentifier函数,没有在实例化一个tableViewCell对象。

    这里也对这个进行了一定的说明:

    /* Retrieve a cell with the given identifier from the table view. The cell is defined in the main storyboard: its identifier is MyIdentifier, and its selection style is set to None. */
    再加上苹果官方TableView Guide文档中写到:

    If the dequeueReusableCellWithIdentifier: method asks for a cell that’s defined in a storyboard, the method always returns a valid cell. If there is not a recycled cell waiting to be reused, the method creates a new one using the information in the storyboard itself. This eliminates the need to check the return value for nil and create a cell manually. 

    就是用dequeueReusableCellWithIdentifier:MyIdentifier函数会一直得到一个有效的在storyboard中定义的cell,而且用storyboard中所定义的信息去设置这个cell,这样就省去了每次去检测cell是否是nil,再去手动的创建。

    从这里可以看到,我的代码中出现的错误就是在storyboard中没有将cell设置为subtitle的样式,否则应该不会出错。

    所以下面这种方式是最简单的:

    var identifer: String = "myCell"
    var cell = tableView.dequeueReusableCellWithIdentifier(identifer) as UITableViewCell
    cell.textLabel.text = a[indexPath.row].name
    cell.detailTextLabel.text = "detail"

    但是要注意,这样方式需要在storyboard中设置好cell的一些必要的属性。

    如果是这样,那自定义cell又该如何实现呢?

    加个你custom的cell类的名字是MyTableViewCell,则实现方法如下

    - (UITableViewCell *)tableView:(UITableView *)tableView
    cellForRowAtIndexPath:(NSIndexPath *)indexPath
    {
        MyTableViewCell *cell = [tableView
    dequeueReusableCellWithIdentifier:@"MyIdentifier"];
        cell.firstLabel.text = [NSString stringWithFormat:@"%d", indexPath.row];
        cell.secondLabel.text = [NSString stringWithFormat:@"%d", NUMBER_OF_ROWS -
    indexPath.row];
        return cell;
    }

    这个是官方提供的OC代码,将其转换成Swift代码就可以了。如下

        func tableView(tableView: UITableView!, cellForRowAtIndexPath indexPath: NSIndexPath!) -> UITableViewCell!{
            var cell: MyTableViewCell = tableView.dequeueReusableCellWithIdentifier("cell") as MyTableViewCell
            cell.first.text = "1111"
            cell.second.text = "2222"
                    
            return cell
        
        }
  • 相关阅读:
    Selenium生成Report的利器- ExtentReports
    学习使用monkey 测试
    charles 结合mocky 模拟数据
    Vue.use()源码分析且执行后干什么了
    commonjs 与 es6相关Module语法的区别
    vue函数化组件 functional
    html5细线表格制作
    移动端H5页面禁止长按复制和去掉点击时高亮
    javascript生成器
    promise和生成器的结合
  • 原文地址:https://www.cnblogs.com/scaptain/p/3948746.html
Copyright © 2011-2022 走看看