2

我有多个 struct timespec 值和一个 time_t 值。我想看看哪个 struct timespec 值最接近 time_t 值。我应该只比较秒而忽略纳秒吗?或者我应该将两者都转换为自纪元以来的总纳秒?或者也许将两者都转换为自 Epoch 将半秒添加到 time_t 以来的总纳秒?

背景: struct timespec 有两个成员:.tv_sec 包含自 Epoch 以来的秒数,.tv_nsec 包含额外的纳秒。time_t 只是秒。在我的例子中, struct timespec 值来自文件系统属性,而 time_t 值是从本地时间字符串转换而来的。我不知道系统调用是否获取本地时间以纳秒为单位或截断它们,所以我不知道,假设 time_t 可能与同时生成的 struct timespec 相差一秒或一秒。

我可以将每个 struct timespec 值中的 .tv_sec 与 time_r 值进行比较,或者我可以将每个 tv_sec 乘以 1,000,000,000 并将其添加到其 tv_nsec,然后将 time_t 乘以 1,000,000,000,然后进行比较?或者还要给time_r增加半秒?自大纪元以来转换为纳秒是否值得麻烦?另外,我应该使用什么类型?无符号长长?

(我不知道我是想太多了还是想得太少了。)

4

1 回答 1

1

不应该关注当地时间,因为time_t(并且.tv_sec也是time_t)与时区无关。它们都指的是自一个共同时代以来的时间。

使用timespec. 使用 astruct timespec来跟踪差异以消除对足够范围的担忧并获得最佳答案。

未经检查的代码,但足以给 OP 一个想法。

#include <time.h>

// Reasonable to assume .tv_nsec is in the normal range [0-999999999]
// and .tv_sec >= 0.
// Else add code to handle -1 .tv_sec and out of range ones.

// Absolute value of the difference.
static struct timespec abs_timespec(struct timespec t, time_t ref) {
  if (t.tv_sec >= ref) {
    t.tv_sec -= ref;
  } else {
    t.tv_sec = ref - t.tv_sec;
    if (t.tv_nsec > 0) {
      t.tv_nsec = 1000000000 - t.tv_nsec;
      t.tv_sec--;
    }
  }
  return t;
}

const struct timespec* compare(size_t n, const struct timespec *t, time_t ref) {
  if (n == 0) {
    return NULL;
  }

  const struct timespec *closest = &t[0];
  struct timespec best_diff = abs_timespec(t[0], ref);

  for (size_t i = 1; i < n; i++) {
    struct timespec diff = abs_timespec(t[i], ref);
    if (diff.tv_sec <= best_diff.tv_sec
        && (diff.tv_sec < best_diff.tv_sec || diff.tv_nsec < best_diff.tv_nsec)) {
      best_diff = diff;
      closest = &t[i];
    }
  }
  return closest;
}
于 2021-02-04T03:57:44.973 回答