记录一次时间单位判断

前端在处理时间戳的时候,通常情况下都是处理 ms,但是由于历史原因不可描述的原因有些脏数据使用了 s。(希望你不要遇到我们这种场景)

怎么办?我们就需要判断下时间戳的单位?

从下面这段代码看出来,1698153828372 可能是 ms,也可能是 s,只是后者是 55782 年的事情了。

scss 复制代码
import dayjs from 'dayjs';

const time = Date.now(); // 1698153828372

// 如果 time 是 ms
dayjs(time).format(); // 2023-10-24T21:22:55+08:00

// 如果 time 是 s
dayjs(time * 1000).format(); // 55782-05-09T05:16:39+08:00

再看一段代码。

scss 复制代码
import dayjs from 'dayjs';

const time = dayjs('1970-02-01').valueOf(); // 2649600000

// 如果 time 是 ms
dayjs(time).format(); // 1970-02-01T00:00:00+08:00

// 如果 time 是 s
dayjs(time * 1000).format(); // 2053-12-18T00:00:00+08:00

也就是说没法通过时间戳判断单位,因为可能是 ms 也可能是 s,都是合理的。


然而实际使用过程中,我们知道不太可能使用很后面的时间,假设最大是 3000 年。

ini 复制代码
// 正常不太可能出现 3000 年以后的日期。所以假定最大时间
const maxDay = baseTime('3000-01-01');
// ms 最大值。 32503651200000
const maxMs = maxDay.valueOf();
// s  最大值。 32503651200
const maxS = maxMs / 1000;
// dayjs(maxS).format(); // 1971-01-12T12:47:31+08:00

也就是 time 如果 > maxS,则就能准确判断其一定为 ms,如果 < maxS,则可能是 ms 或者 s。

于是我们可以得到这段代码。通过 guessTimeUnit 识别出一定是 ms。但是判断不出 s 还是 ms。

好在这个 maxS 的界限在于 1971-01-12T12:47:31+08:00,也就意味着准确判断 ms 的概率非常大,我认为这个概率在 99% 了。

typescript 复制代码
import { isNumber } from 'lodash-es';

// 正常不太可能出现 3000 年以后的日期。所以假定最大时间
const maxDay = baseTime('3000-01-01');
// ms 最大值。 32503651200000
const maxMs = maxDay.valueOf();
// s  最大值。 32503651200
const maxS = maxMs / 1000;
// dayjs(maxS).format(); // 1971-01-12T12:47:31+08:00

type Result = 'ms' | undefined;

/**
 * 猜测时间单位,因为有假设不太可能出现 3000,所以带前缀 guess
 */
function guessTimeUnit(time: number /* s or ms */): Result {
  // 判断是数值
  if (isNumber(time)) {
    if (time >= maxS) {
      // 大于 maxS, 可能是 ms
      return 'ms';
    }
  }

  // 可能是 s or ms,判断不了,返回 undefined,交由调用方判断
  return;
}

这基本能解决很多问题了,如果假定业务极小概率出现 1971-01-12T12:47:31+08:00 之前的 ms。那么 time < maxS 就认定为 s。

当然这是一种假设,实际还是会发生的,程序需要严谨对待。


如果我们处于特定场景,可以就可以忽略小概率的事情。

比如函数 ms2s,正常入参是 ms,由于历史原因可能也传入了 s。这个时候可以这样应对。

1 能判断时间单位则按照判断的走。

2 如果不能判断(小概率),这按照函数的定义走。 ms2s 定义传入 ms,所以按照 ms 走。

sql 复制代码
const THOUSAND = 1000;

/** 可能传入的 time 没有按照要求,做个兼容 */
function compatible(unit, method) {
  /** time: s or ms */
  function compatibleMethod(time: number): number {
    // 做好判断
    if (isNumber(time)) {
      // 如果能判断出是毫秒,则做兼容
      if (isMs(time)) {
        if (unit === 's') {
          return Math.floor(time / THOUSAND);
        }
        if (unit === 'ms') {
          return Math.floor(time);
        }
      }

      // 是毫秒或者秒
      return method(time);
    }

    // 非 number, 可能 null undefined, 可能 string,没法处理,原样返回
    return time;
  }

  return compatibleMethod;
}

const ms2s = compatible('s', (time: number) => {
  return Math.floor(time / THOUSAND);
});
相关推荐
天涯学馆2 小时前
Next.js与NextAuth:身份验证实践
前端·javascript·next.js
HEX9CF2 小时前
【CTF Web】Pikachu xss之href输出 Writeup(GET请求+反射型XSS+javascript:伪协议绕过)
开发语言·前端·javascript·安全·网络安全·ecmascript·xss
ConardLi2 小时前
Chrome:新的滚动捕捉事件助你实现更丝滑的动画效果!
前端·javascript·浏览器
ConardLi2 小时前
安全赋值运算符,新的 JavaScript 提案让你告别 trycatch !
前端·javascript
积水成江3 小时前
关于Generator,async 和 await的介绍
前端·javascript·vue.js
Z3r4y3 小时前
【Web】portswigger 服务端原型污染 labs 全解
javascript·web安全·nodejs·原型链污染·wp·portswigger
人生の三重奏3 小时前
前端——js补充
开发语言·前端·javascript
Tandy12356_3 小时前
js逆向——webpack实战案例(一)
前端·javascript·安全·webpack
老华带你飞3 小时前
公寓管理系统|SprinBoot+vue夕阳红公寓管理系统(源码+数据库+文档)
java·前端·javascript·数据库·vue.js·spring boot·课程设计
qbbmnnnnnn4 小时前
【WebGis开发 - Cesium】如何确保Cesium场景加载完毕
前端·javascript·vue.js·gis·cesium·webgis·三维可视化开发