2019最新iOS面试(OC+swift)iOS 进阶IOS&Swift

『ios』多个cell时间倒计时,最佳实现方法探索。

2018-11-18  本文已影响180人  butterflyer
image.png

最近想起来以前看一个项目时的需求,是一个商品页的倒计时功能,当然是每个cell上都有一个倒计时。于是进行了一番探索。目的在于找到最恰当的方法,耗性能最低的方法来实现这个需求。

如果相对nstimer有更多了解可以看我之前整理的这篇文章,ios』来自NSTimer的坑 ,告别循环, 最全的方法总结

三种方法的总结demo在这里

第一种方法

我觉得这是最容易想到的方法,也是耗性能最高的方法。每个cell上搞一个时间计时器nstimer,当然是有复用的,但是这也会创建1个以上的nstimer。
暴露下面的方法给外界调用,我们会发现,虽然并不会每个cell都会去创建nstimer,但是也确实会创建一个以上的数量。

//开始计时
- (void)coundDownStart:(NSInteger)time
{
    currentTime = time;
    [self show];
    if (!coundTimer)
    {
        coundTimer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(coundDown:) userInfo:nil repeats:YES];

        [[NSRunLoop currentRunLoop]addTimer:coundTimer forMode:NSRunLoopCommonModes];
    }
}

有创建就会有释放,在页面将要消失的时候,我们拿到cell中nstimer所在的数组,然后全部释放掉,就可以。

- (void)killTimer
{
    if (self.timerArray)
    {
        for (NSTimer *subTimer in self.timerArray)
        {
            if (subTimer)
            {
                [subTimer invalidate];
            }
        }
        [self.timerArray removeAllObjects];
    }
}

第二种方法

上面一个种方法是利用cell的复用,创建了nstimer,数量是多个的,现在我们要考虑的肯定是想办法只用一个nstimer来实现这个需求。
我们用这里创建一个时间计时器,然后每秒钟都去遍历,这个数据源时间,然后根据NSIndexPath拿到所有的cell,挨个赋值。这也是一种实现方式。

- (void)countdown{
    
    NSInteger time;
    
    for (int i = 0; i < _showTimes.count; i++) {
        
        time = [_showTimes[i][showTimeKey] integerValue];
        
        NSIndexPath *indexPath = [NSIndexPath indexPathForRow:[_showTimes [i][indexPathKey] integerValue] inSection:0];
        
        UITableViewCell *cell = [_myTableView cellForRowAtIndexPath:indexPath];
        
        if (time > 0) {
            cell.textLabel.text =[NSString stringWithFormat:@"活动结束还剩下:%ld 秒", time];
            time --;
            NSDictionary *dic = @{indexPathKey: [NSString stringWithFormat:@"%d",I],
                                  showTimeKey: [NSString stringWithFormat:@"%ld",time]};
            
            [_showTimes replaceObjectAtIndex:i withObject:dic];
        }else {
            cell.textLabel.text = [NSString stringWithFormat:@"活动已结束"];   
        }   
    }  
}

最好的方法,第三种方法。

上面虽然是创建了一个时间计时器,但是还不够,因为每次都会去遍历这个时间数组,如果这个数据源很大的话,效果肯定也不理想。
所以我们可以这样,同样,创建一个计时器,然后去遍历这个tableView中目前可见的cell,拿到这些cell,然后分别赋值。
拿出可见cell的api

NSArray  *cells = tableView.visibleCells; //取出屏幕可见cell   

这里我们采用了gcd的形式。
为何要采取gcd的形式呢?
想一下,nstimer依赖于什么,runloop是吧,我们需要注意要启动runloop,而且还需要注意mode的种类,用commonmodel,防止在滑动tableView的时候,timer停止。
还有,我们还需要注意之前说的,nstimer防止循环引用的问题。
还有,NSTimer的创建与撤销必须在同一个线程操作。~

合理利用了cell.tag值跟cell.timeLabel的tag值来区分

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    BestCell * cell = [tableView dequeueReusableCellWithIdentifier:@"BestCell"];
    
    /// 3.绑定tag
    cell.tag = indexPath.row;
    cell.timeLabel.tag = 1314;
    return cell;
}
 if (_timer==nil) {
        dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
        _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0,queue);
        dispatch_source_set_timer(_timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0); //每秒执行
        dispatch_source_set_event_handler(_timer, ^{
            dispatch_async(dispatch_get_main_queue(), ^{
                
                NSArray  *cells = tableView.visibleCells; //取出屏幕可见ceLl
                
                for (UITableViewCell * cell in cells) {
                    
                    NSDate * sjDate = [NSDate date];   //手机时间
                    NSInteger sjInteger = [sjDate timeIntervalSince1970];  // 手机当前时间戳
                    NSString* tempEndTime;
                    if ([dataList[0] isKindOfClass:[NSArray class]]) {
                        NSInteger section = cell.tag / 1000;
                        NSInteger row = cell.tag % 1000;
                        tempEndTime = dataList[section][row];
                    }else {
                        
                        tempEndTime = dataList[cell.tag];
                    }
                    
                    for (UIView * labText in cell.contentView.subviews) {
                        if (labText.tag == 1314) {
                            UILabel * textLabel = (UILabel *)labText;
                            NSInteger endTime = tempEndTime.longLongValue + _timeLess;
                            
                            textLabel.text = [self getNowTimeWithString:endTime :sjInteger];
                        }
                    }
   
                }
            });
        });
        dispatch_resume(_timer); // 启动定时器
    }

取消计时器的话,调用cancel方法就可以了,是不是很easy。

/**
 *  主动销毁定时器
 */
-(void)destoryTimer{
    if (_timer) {
        dispatch_source_cancel(_timer);
        _timer = nil;
    }
}

想要找一群一起进步的人,请加入我

image.png
上一篇下一篇

猜你喜欢

热点阅读