demo地址
效果图
1.时间处理
既然是倒计时,肯定要有一个对比时间,所以为了保证活动倒计时不受修改本机系统时间的影响,参考对比时间为服务器时间。
获取服务器时间的方式是在项目的请求管理类中解析到每一次请求的header信息,里面一般包含一个时间戳
NSHTTPURLResponse *resp = (NSHTTPURLResponse *)response;
NSDictionary *allHeaders = resp.allHeaderFields;
NSString *dateServer = [allHeaders objectForKey:@"Date"];
例如demo中的allHeaderFields
打印如下:
{
"Content-Encoding" = gzip;
"Content-Length" = 1145;
"Content-Type" = "text/html";
Date = "Wed, 15 Apr 2020 01:58:59 GMT";
Server = bfe;
}
- 需要做的就是存储每次请求响应中的这个时间戳,demo封装好了分类
NSDate+InternetDateTime
,可以直接调用并存储最新的时间,以便下次使用
// 记录服务器时间
NSDate *inputDate = [NSDate dateFromInternetDateTimeString:dateServer formatHint:DateFormatHintRFC822];
if (inputDate) {
[NSDate saveServerTime:inputDate];
}
注意: 计算服务器时间戳的原理就是将请求响应的服务器时间戳和当前本机系统时间戳同时存储下来。在需要使用服务器时间戳时,同时取出之前存储的两个时间戳,计算得到一个差值。然后再获取本机系统时间戳,和差值做计算,得到一个比较准确的服务器时间戳,这个时间戳才是我们倒计时中的参考时间了。
另外还有一点需要注意的是要监听系统时间发生变化,当前程序正在运行期间去改变了系统时间的话,要及时的更新当前时间的计算。可在AppDelegate中实现以下方法
/// 监听系统时间发生变化
[[NSNotificationCenter defaultCenter] addObserver:object selector:@selector(systemClockChange:) name:NSSystemClockDidChangeNotification object:nil];
/// 监听系统时间发生变化时,更新存储的本地时间
-(void)systemClockChange:(id)sender{
double currentLocalTimeStamp = [[NSDate date] timeIntervalSince1970] * 1000;
[[NSUserDefaults standardUserDefaults] setDouble:currentLocalTimeStamp forKey:kLastSaveLocalTimeStamp];
[[NSUserDefaults standardUserDefaults] synchronize];
}
2.倒计时处理
封装一个倒计时管理类
SYCountDownManager
,内部创建一个计时器,通过传入的截止时间,直接生成倒计时时间,并回传时间模型。时间模型中提供了int
和NSString
两种数据类型。也可通过SYCountDownManager
生成固定格式的整体倒计时字符串,例如:9:29:31
关于计时器的选择,没有实际测验,但根据网上查找到的资料分析发现GCD更好些,所有选取了GCD实现计时器。也可以通过其它的方式,只需替换相关代码即可
计时器的销毁,可根据实际情况调用
destoryTimer
。
3.列表处理
- 使用过程中发现如果列表中同时存在多个倒计时,滚动页面时倒计时会在开始出现时有突然的跳动,为了处理这个问题需要在cell将要出现时就提前给控件赋值。如下两个系统方法中:
- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath;
- (void)collectionView:(UICollectionView *)collectionView willDisplayCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath