iOS NSDate准确吗?精确获取时间
在APP开发中我们经常涉及到时间这一块,在请求时服务端数据时,有时候会带上时间戳,因为客户端的时间不是很靠谱(用户随时可能自己修改时间),所以原则上,客户端的时间仅仅作为参考。
NSDate
NSDate是我们平时使用较多的一个类,它描述的是时间线上的一个绝对的值,和时区和文化无关,它参考的值是:以UTC为标准的,2001年1月1日00:00:00这一刻的时间绝对值。
常见用法
NSDate *date = [NSDate date];
上面的date输出的是绝对的UTC时间,而北京时间的时区为UTC+8,上面的输出+8个小时,就是当前的时间了。
NSDate和市区和文化无关,所以要展示具体格式的时间,我们需要NSDateFormatter和NSTimeZone的辅助。NSDate是受手机系统时间控制的,用户可能会修改它的值,所以NSDate并不可靠。
CFAbsoluteTimeGetCurrent()
CFAbsoluteTimeGetCurrent()属于CoreFoundation,概念和NSDate非常相似,只不过参考点是:以GMT为标准的,2001年1月1日00:00:00这一刻的时间绝对值。它相当于[[NSDate date] timeIntervalSinceReferenceDate]
,它也是不可靠的的,也会跟着当前设备的系统时间一起变化。
gettimeofday
gettimeofday获得的值是Unix time,Unix time是以UTC 1970年1月1号 00:00:00为基准时间,当前时间距离基准点偏移的秒数。NSDate也有一个API能返回Unix time:
NSDate *date = [ NSDate date];
NSLog( @"timeIntervalSince1970: %f", [date timeIntervalSince1970]);
gettimeofday的用法如下
struct timeval now;
struct timezone tz;
gettimeofday(&now, &tz);
NSLog(@"gettimeofday: %ld",now.tv_sec);
gettimeofday和NSDate,CFAbsoluteTimeGetCurrent()一样,都是受当前设备的系统时间影响。只不过是参考的时间基准点不一样而已。我们和服务器通讯的时候一般使用Unix time。
mach_absolute_time()
mach_absolute_time()返回的就是CPU已经运行的CPU的时钟周期数(ticks)。tick的数量可以用来描述时间,将这个tick数经过一定的转换就可以变成秒数,或者纳秒数,这样就和时间直接关联了。
不过这个tick数,在每次手机重启之后,会重新开始计数,而且iPhone锁屏进入休眠之后tick也会暂停计数。
mach_absolute_time()不会受系统时间影响,只受设备重启和休眠行为影响。
#import <mach/mach_time.h>
uint64_t getTickCount(void)
{
static mach_timebase_info_data_t sTimebaseInfo;
uint64_t machTime = mach_absolute_time();
// Convert to nanoseconds - if this is the first time we've run, get the timebase.
if (sTimebaseInfo.denom == 0 )
{
(void) mach_timebase_info(&sTimebaseInfo);
}
// 得到毫秒级别时间差
uint64_t millis = ((machTime / 1e6) * sTimebaseInfo.numer) / sTimebaseInfo.denom;
return millis;
}
CACurrentMediaTime()
CACurrentMediaTime()属于QuartzCore,CACurrentMediaTime就是将上面mach_absolute_time()的CPU tick数转化成秒数的结果。
#import <QuartzCore/QuartzCore.h>
double mediaTime = CACurrentMediaTime();
sysctl
sysctl函数的定义
int sysctl(int *, u_int, void *, size_t *, void *, size_t);
- sysctl的第一参数是个数组,按照顺序第一个元素指定本请求定向到内核的哪个子系统,第二个及其后元素依次细化指定该系统的某个部分,类推...
- 之后是数组的长度u_int.
- 后面的size_t *需要注意,当sysctl被调用时,size指向的值指定该缓冲区的大小;函数返回时,该值给出内核存放在该缓冲区中的数据量,如果这个缓冲不够大,函数就返回ENOMEM错误
- sysctl函数的结果:成功:返回0; 失败:返回-1
获取时间方法为
#include <sys/sysctl.h>
// 手机运行时长
- (time_t)uptime {
struct timeval boottime;
int mib[2] = {CTL_KERN, KERN_BOOTTIME};
size_t size = sizeof(boottime);
time_t now;
time_t uptime = -1;
(void)time(&now);
if (sysctl(mib, 2, &boottime, &size, NULL, 0) != -1 && boottime.tv_sec != 0)
{
uptime = now - boottime.tv_sec;
}
return uptime;
}
返回的值是上次设备重启的Unix time。
这个API返回的值也会受系统时间影响,用户如果修改时间,值也会随着变化。
NSDate、CFAbsoluteTimeGetCurrent()常用于日常时间、时间戳的表示,与服务器之间的数据交互
CACurrentMediaTime() 常用于测试代码的效率。