ios - 为什么NSDate的dateWithTimeIntervalSince1970修改其输入?

标签 ios objective-c nsdate

我正在运行以下代码:

for (int i = 0; i < 100; ++i) {
  NSDate *date = [NSDate date];
  NSTimeInterval interval = date.timeIntervalSince1970;
  NSDate *newDate = [NSDate dateWithTimeIntervalSince1970:interval];

  if (![date isEqualToDate:newDate]) {
    NSLog(@"Not equal!");
  }
}

令人惊讶的是,在许多迭代中,日期彼此并不相等。怎么会这样?

最佳答案

反汇编dateWithTimeIntervalSince1970:显示它调用-initWithIntervalSinceReferenceDate::

CoreFoundation`+[NSDate dateWithTimeIntervalSince1970:]:
    0x10fd39430 <+0>:  pushq  %rbp
    0x10fd39431 <+1>:  movq   %rsp, %rbp
    0x10fd39434 <+4>:  pushq  %rbx
    0x10fd39435 <+5>:  pushq  %rax
    0x10fd39436 <+6>:  movsd  %xmm0, -0x10(%rbp)
    0x10fd3943b <+11>: movq   0x2d8146(%rip), %rsi      ; "alloc"
    0x10fd39442 <+18>: movq   0x29edc7(%rip), %rbx      ; (void *)0x000000010f35e940: objc_msgSend
    0x10fd39449 <+25>: callq  *%rbx
    0x10fd3944b <+27>: movsd  -0x10(%rbp), %xmm0        ; xmm0 = mem[0],zero 
    0x10fd39450 <+32>: addsd  0x1f9d58(%rip), %xmm0     ; _CFLog_os_trace_type_map + 16
    0x10fd39458 <+40>: movq   0x2d9041(%rip), %rsi      ; "initWithTimeIntervalSinceReferenceDate:"
    0x10fd3945f <+47>: movq   %rax, %rdi
    0x10fd39462 <+50>: callq  *%rbx
    0x10fd39464 <+52>: movq   0x2d81b5(%rip), %rsi      ; "autorelease"
    0x10fd3946b <+59>: movq   %rax, %rdi
    0x10fd3946e <+62>: movq   %rbx, %rax
    0x10fd39471 <+65>: addq   $0x8, %rsp
    0x10fd39475 <+69>: popq   %rbx
    0x10fd39476 <+70>: popq   %rbp
    0x10fd39477 <+71>: jmpq   *%rax
    0x10fd39479 <+73>: nopl   (%rax)

此外,初始化程序的输入为 secs - 978307200(或 secs - NSTimeIntervalSince1970),这是 1970 年与引用日期之间的时间差。此计算在从 double 中减去整数( double )时,可能会由于舍入误差而更改输入值的分数。例如,这是测试失败的日期:

date: Sun Mar 25 17:54:39 2018 (543682479.8504179716),
newDate: Sun Mar 25 17:54:39 2018 (543682479.8504180908)

由于 log2(543682479.8504179716 + NSTimeIntervalSince1970) ~ 30.5log2(543682479.8504179716) ~ 29.01,需要调整 double 值的指数,并且需要调整尾数归一化,这可能会影响分数值。

解决方案是使用 +dateWithTimeIntervalSinceReferenceDate: 工厂方法,它直接初始化一个 NSDate 而无需额外的计算。

关于ios - 为什么NSDate的dateWithTimeIntervalSince1970修改其输入?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49480305/

相关文章:

iphone - iPad上的黑屏而不是启动图像

ios - 是否可以在没有相机的情况下使用 Vuforia?

objective-c - cocoa 无法连接 Action

ios - Swift 将 NSDate 转换为 24h 格式在设备上失败

ios - 错误的时间...从时间戳开始

ios - 无法在 swift 中获得回调?

ios - 按值 Swift 将数组拆分为数组

html - 以编程方式调整 wkwebview 的高度

ios - NSURL - 获取错误消息

objective-c - 仅获取两个日期之间的周末