iOS開發之UITableViewCell可暫停倒計時

前言

最近開發中,用到了UITableViewCell倒計時功能,這裏將這部分功能分離出來,供你們參考。git

1.原理

考慮到APP性能,這裏只建立一個定時器,定時刷新當前正在顯示的UITableViewCell,使用Model記錄剩餘倒計時時間和當前UITableViewCell是否暫停。github

2.核心代碼

建立定時器

考慮到方便和不須要銷燬,這裏定時器使用GCD--->GCD定時器封裝OC&Swift bash

self.timer = [[CLGCDTimer alloc] initWithInterval:1 delaySecs:0 queue:dispatch_get_main_queue() repeats:YES action:^(NSInteger actionTimes) {
                __typeof(&*weakSelf) strongSelf = weakSelf;
                strongSelf.actionTimes = actionTimes;
                [strongSelf reloadVisibleCells];
            }];
            [self.timer start];
複製代碼

刷新當前正在顯示的UITableViewCell

這裏只對正在顯示的UITableViewCell進行操做,找出當前正在顯示的UITableViewCell對應的數據Model,對數據源進行修改後刷新UITableViewapp

- (void)reloadVisibleCells {
    for (CLCountdownCell *cell in self.tableView.visibleCells) {
        NSIndexPath *indexPath = [self.tableView indexPathForCell:cell];
        CLCountdownModel *model = [self.arrayDS objectAtIndex:indexPath.row];
        model.actionTimes = self.actionTimes;
        model.leaveTime = self.leaveTime;
        if (model.isPause) {
            continue;
        }
        cell.model = model;
    }
}
複製代碼

Model數據修正

由於只修改了當前正在顯示的UITableViewCell對應Model的數據源,因此滑動出來的UITableViewCell對應Model數據源並不正確,這裏在UITableViewCell即將顯示的時候修正對應數據源。性能

- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath {
    CLCountdownModel *model = [self.arrayDS objectAtIndex:indexPath.row];
    model.actionTimes = self.actionTimes;
    model.leaveTime = self.leaveTime;
    if (!model.isPause) {
        CLCountdownCell *countdownCell = (CLCountdownCell *)cell;
        countdownCell.model = model;
    }
}
複製代碼

記錄暫停狀態

經過修改對應UITableViewCell所在數據Model,記錄當前UITableViewCell暫停狀態,達到暫停效果。這裏須要注意記錄暫停當時的時間以及從新開始的時間。ui

- (void)setIsPause:(BOOL)isPause {
    if (isPause) {
        self.pauseTime = self.remainingTime;
        self.startTime = 0;
    }else {
        _isPause = isPause;
        self.startTime = self.remainingTime;
    }
    _isPause = isPause;
}
- (NSInteger)remainingTime {
    if (_isPause) {
        return self.pauseTime;
    }else {
        if (self.pauseTime != 0 && self.startTime != 0) {
            return self.countdownTime - self.actionTimes + self.pauseTime - self.startTime - self.leaveTime;
        }else {
            return self.countdownTime - self.actionTimes - self.leaveTime;
        }
    }
}
複製代碼

APP進入後臺記錄

當APP進入後臺,須要記錄當前時間,當再次進入前臺的時候,須要減去離開時間,這樣即便進入後臺也不會影響到倒計時,這裏考慮到進入後臺期間修改時間等操做,直接使用系統運行時間進行記錄。spa

///系統當前運行了多長時間
///由於兩個參數都會受用戶修改時間的影響,所以它們想減的值是不變的
+ (NSTimeInterval)uptimeSinceLastBoot {
    //獲取當前設備時間時間戳 受用戶修改時間影響
    struct timeval now;
    struct timezone tz;
    gettimeofday(&now, &tz);
    
    //獲取系統上次重啓的時間戳 受用戶修改時間影響
    struct timeval boottime;
    int mib[2] = {CTL_KERN, KERN_BOOTTIME};
    size_t size = sizeof(boottime);
    
    double uptime = -1;
    if (sysctl(mib, 2, &boottime, &size, NULL, 0) != -1 && boottime.tv_sec != 0) {
        //獲取上次啓動時間成功
        //秒
        uptime = now.tv_sec - boottime.tv_sec;
        //微秒
        uptime += (double)(now.tv_usec - boottime.tv_usec) / 1000000.0;
    }
    return uptime;
}
複製代碼

監聽APP進入後臺和進入前臺通知,進行記錄。code

- (void)addNotification {
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationWillResignActive:)
                                                 name:UIApplicationWillResignActiveNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationDidBecomeActive:)
                                                 name:UIApplicationDidBecomeActiveNotification object:nil];
}
- (void)applicationWillResignActive:(NSNotification *)notification {
    self.resignSystemUpTime = [NSDate uptimeSinceLastBoot];
    [self.timer suspend];
}
- (void)applicationDidBecomeActive:(NSNotification *)notification {
    self.becomeSystemUpTime = [NSDate uptimeSinceLastBoot];
    self.leaveTime += (NSInteger)floor(self.becomeSystemUpTime - self.resignSystemUpTime);
    [self.timer resume];
}
複製代碼

3.效果圖

這裏數據源建立了10萬,由於只有一個定時器,而且只刷新當前正在顯示的UITableViewCell,因此滑動起來並不會有任何卡頓。 cdn

效果圖

4.總結

UITableViewCell倒計時代碼並很少,只是須要注意一些細節,記錄對應時間和狀態。更多細節請參考文章對應Demo---->CLDemo若有幫助,歡迎Star。server

相關文章
相關標籤/搜索