ios 倒计时列表的性能优化

作者: 嘿晴天 | 来源:发表于2015-12-29 18:15 被阅读4327次

在最近的开发中遇到了一个需求,要在tableview 揭晓接下来正要展示的倒计时展示即将揭晓的物品,倒计时展示类似这个demo
demo 地址: https://github.com/heysunnyboy/leftTimeDemo.git

time.gif

(关键词:倒计时)听到这个上面的这个需求,第一点你会想到什么了,定时器没错,接下来是cell 里面放定时器 吗,那么如果是这样的话会产生什么样的后果呢,
我一开始也是这样想的,然后tableview 显示出来没多久,由于创建的定时器过多程序就卡死了,cell 的高度100左右 ,在iphone 6的屏幕大致创建 至少6个定时器左右(只是在屏幕内的cell 创建定时器)这个足以导致 性能下降

那么让我们尝试使用一个定时器来完成所有的倒计时操作,既然要往这个方向做,我们先来整理下思路

33471EE3-2679-4001-895F-842834241C31.png

在这里,可能会遇到一个问题,获取屏幕内可见的cell的时候,并不知道当前cell是那个,对应的需要的展示的倒计时,怎么对应上,那么我们可以给cellforindexpath 的方法给cell.tag = indexPath.row 通过 tag 标签 和时间数组的索引,对应找出相应的时间展示的倒计时

//定时器刷新倒计时
-(void)calTime
{
    NSArray  *cells = _tableView.visibleCells; //取出屏幕可见cell
    for (UITableViewCell *cell in cells) {
        cell.textLabel.text = [self getTimeStr:timeArr[cell.tag]];
    }
}
//返回倒计时
-(NSString *)getTimeStr:(NSString *)fireStr
{
    NSDateFormatter* formater = [[NSDateFormatter alloc] init];
    [formater setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
    NSDate* fireDate = [formater dateFromString:fireStr];
    NSDate *today = [NSDate date];
    NSCalendar *calendar = [NSCalendar currentCalendar];
    unsigned int unitFlags = NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit | NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit;
    NSDateComponents *d = [calendar components:unitFlags fromDate:today toDate:fireDate options:0];//计算时间差
    long hour = [d day] *24 + [d hour];
    NSString *seconds;
    NSString *minutes;
    NSString *hours;
    if([d second]<10)
        seconds = [NSString stringWithFormat:@"0%ld",[d second]];
    else
        seconds = [NSString stringWithFormat:@"%ld",[d second]];
    if([d minute]<10)
        minutes = [NSString stringWithFormat:@"0%ld",[d minute]];
    else
        minutes = [NSString stringWithFormat:@"%ld",[d minute]];
    if(hour < 10)
        hours = [NSString stringWithFormat:@"0%ld", hour];
    else
        hours = [NSString stringWithFormat:@"%ld",hour];
    return [NSString stringWithFormat:@"            倒计时%@:%@:%@", hours, minutes,seconds];
}

这里已经将需要展示的倒计时正常展示出来了,那么还有什么可以优化一下,我们还可以对定时器进行简单的优化,就是在页面消失的时候,将定时器置空,停止轮询,在页面出现才可以轮询。

-(void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    if (_timer == nil) {
        _timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(calTime) userInfo:nil repeats:YES];
    }
}
-(void)viewDidDisappear:(BOOL)animated
{
    [super viewDidDisappear:animated];
    if(_timer)
    {
        _timer = nil;//关闭定时器,
    }
}

相关文章

网友评论

  • 移动的WIFI:所以每秒都在刷新TableView?
    嘿晴天:@移动的WIFI 刷新单个cell
  • 9925ee2bf0cc:我想请问下改怎么加上毫秒的计时呢 大神
  • 心至靜行至遠:频繁的创建NSDateFormatter对象也会产生很大的资源消耗,建议楼主缓存一下,对性能也有一定的提升!
    嘿晴天:@心至靜行至遠 好的谢谢哈
  • 松树李树:厉害了我的哥
  • Dolphii:那如果这个有分页和回退功能 就好像不适合了
  • 不舍红尘的和尚松:如果刷新了数据调了tableView reloadDate 你这个timeArr 应该写在哪里呢。reloadDate只有代理方法 timeArr更新了数据怎么办
    timeArr = @[@"2015-12-31 19:23:20",@"2015-12-31 20:23:20",@"2015-12-30 19:23:20",@"2015-12-30 19:23:20",@"2015-12-31 22:23:20",@"2015-12-31 19:23:20",@"2015-12-31 20:23:20",@"2015-12-30 19:23:20",@"2015-12-30 19:23:20",@"2015-12-31 22:23:20",@"2015-12-31 19:23:20",@"2015-12-31 20:23:20",@"2015-12-30 19:23:20",@"2015-12-30 19:23:20",@"2015-12-31 22:23:20",@"2015-12-31 19:23:20",@"2015-12-31 20:23:20",@"2015-12-30 19:23:20",@"2015-12-30 19:23:20",@"2015-12-31 22:23:20"];
    不舍红尘的和尚松:@嘿晴天 好的 我试试 :blush:
    嘿晴天:@不舍红尘的和尚松 写在-(void)calTime
    {
    NSArray *cells = _tableView.visibleCells; //取出屏幕可见cell
    for (UITableViewCell *cell in cells) {
    cell.textLabel.text = [self getTimeStr:timeArr[cell.tag]];
    }
    } 这个方法里
    不舍红尘的和尚松:@不舍红尘的和尚松 请教一下 谢谢
  • 小凡凡520:invaliate
    小凡凡520:@嘿晴天 销毁定时器
    嘿晴天:@小凡凡520 什么?
  • 看个客人:滑动的时候,为什么label上的字符串不会变化
    嘿晴天:@看个客人 是可以的我试过了,你在我写的viewdid load 最后一行加上这个
    [[NSRunLoop currentRunLoop] addTimer:_timer forMode:UITrackingRunLoopMode];
    看个客人:@嘿晴天 好像不行
    嘿晴天:@看个客人 定时器开在主线程runloop了,你滑动的时候主线程去处理界面了,没空管那个定时器。
    可以改成这样NSTimer * timer = [NSTimer scheduledTimerWithTimeInterval:1.0
    target:self
    selector:@selector(haha)
    userInfo:nil
    repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];
  • 癫癫的恋了:还有个优化点,NSDateFormatter的初始化耗时是很高的,可以只实例化一次
    嘿晴天:@癫癫的恋了 好的,get到了谢谢指教
  • 遗编絶简:写的不错。格式化两位字符串可以使用%02d
    嘿晴天:@遗编絶简 好的谢谢你的意见
    遗编絶简:@遗编絶简 就可以减少12行代码了一个format搞定不用判断
  • Delpan:定时器要移除…
    嘿晴天:@Delpan soga还要invalidte
    Delpan:@嘿晴天 timer = nil不代表移除了定时器,定时器添启动后会被强制保留直到被调用暂定函数
    嘿晴天:@Delpan ?
  • Eugene_iOS:不错,学习了
  • IoserT:mark 学习了
  • newbiecoder:学习了
  • 郑钦洪_:醍醐灌顶 讲的特别好
  • RasonWu:一个定时器就搞掂,不错

本文标题:ios 倒计时列表的性能优化

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