最近開發中,用到了UITableViewCell
倒計時功能,這裏將這部分功能分離出來,供你們參考。git
考慮到APP性能,這裏只建立一個定時器,定時刷新當前正在顯示的UITableViewCell
,使用Model
記錄剩餘倒計時時間和當前UITableViewCell
是否暫停。github
考慮到方便和不須要銷燬,這裏定時器使用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
對應的數據Model
,對數據源進行修改後刷新UITableView
。app
- (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;
}
}
複製代碼
由於只修改了當前正在顯示的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進入後臺,須要記錄當前時間,當再次進入前臺的時候,須要減去離開時間,這樣即便進入後臺也不會影響到倒計時,這裏考慮到進入後臺期間修改時間等操做,直接使用系統運行時間進行記錄。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];
}
複製代碼
這裏數據源建立了10萬,由於只有一個定時器,而且只刷新當前正在顯示的UITableViewCell
,因此滑動起來並不會有任何卡頓。 cdn
UITableViewCell
倒計時代碼並很少,只是須要注意一些細節,記錄對應時間和狀態。更多細節請參考文章對應Demo---->CLDemo若有幫助,歡迎Star。server