记一次神奇的时间转换问题(SheetJS)

2023-05-16

最近在写一个功能,使用SheetJS读取Excel表格,在读取日期的时候发现了一个隐藏很深的坑,特此记录一下。
SheetJS读取Excel文件时,可指定参数 cellDates: true,这样当单元格内存储的为日期时,可以将读取到的值直接转化为Date对象。

const workbook = XLSX.read(buffer, { type: 'binary',cellDates: true });

但测试时发现,读取到的值和实际时间总有43秒的误差:
Excel中的值
实际读取到的值
经过搜索,发现这是一个历史原因导致的BUG:

在当地时间1901-01-01 00:00:00,上海时区由LMT(Local Mean Time)切换为CST (China Standard Time),与GMT的时差由+8:05:43调整为+8:00:00。
https://zhuanlan.zhihu.com/p/89914219

在GitHub上也有相关的讨论
GitHub: The exported date is 43 seconds longer

SheetJS用 getTimezoneOffset 来解决这个问题,但 getTimezoneOffset 获取到的时差只精确到分钟,所以就产生了43秒的误差

这个Issue里给出了解决方法

方法1

手动修复误差

function getTimezoneOffsetMS(date) {
  var time = date.getTime();
  var utcTime = Date.UTC(date.getFullYear(),
                         date.getMonth(),
                         date.getDate(),
                         date.getHours(),
                         date.getMinutes(),
                         date.getSeconds(),
                         date.getMilliseconds());
  return time - utcTime;
}

const importBugHotfixDiff = (function () {
  const basedate = new Date(1899, 11, 30, 0, 0, 0);
  const dnthreshAsIs = (new Date().getTimezoneOffset() - basedate.getTimezoneOffset()) * 60000;
  const dnthreshToBe = getTimezoneOffsetMS(new Date()) - getTimezoneOffsetMS(basedate);
  return dnthreshAsIs - dnthreshToBe;
}());

function fixPrecisionLoss(date) {
  return (new Date(date.getTime() - importBugHotfixDiff));
}
方法2

SSF 模块对 Date 对象进行转换

import XLSX from 'xlsx';
const basedate = new Date(1899, 11, 30, 0, 0, 0);
const dnthresh = basedate.getTime() + (new Date().getTimezoneOffset() - basedate.getTimezoneOffset()) * 60000;

const day_ms = 24 * 60 * 60 * 1000;
const days_1462_ms = 1462 * day_ms;

function datenum(v: Date, date1904: boolean) {
  let epoch = v.getTime();
  if (date1904) {
    epoch -= days_1462_ms;
  }
  return (epoch - dnthresh) / day_ms;
}

export function fixImportedDate(date: Date, isDate1904: boolean) {
  const parsed = XLSX.SSF.parse_date_code(datenum(date, false), { date1904: isDate1904 });
  // return `${parsed.y}-${parsed.m}-${parsed.d}`;
  return new Date(parsed.y, parsed.m, parsed.d, parsed.H, parsed.M, parsed.S);
}
其他方法

不使用cellDates: true,直接用SSF模块转换日期格式单元格里的值

import XLSX from 'xlsx';

export function parseExcelDate(n: number, isDate1904: boolean) {
  const parsed = XLSX.SSF.parse_date_code(n, { date1904: isDate1904 });
  // return `${parsed.y}-${parsed.m}-${parsed.d}`;
  return new Date(parsed.y, parsed.m, parsed.d, parsed.H, parsed.M, parsed.S);
}

后记

真是醉了,写个代码还能学到历史……

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

记一次神奇的时间转换问题(SheetJS) 的相关文章

  • ROS-gazebo 简介

    gazebo 一 什么是Gazebo xff1f Gazebo的典型用途Gazebo的一些主要特点二 系统要求三 安装 1 添加源2 安装Gazebo四 用户界面 1 场景2 左右面板3 工具栏4 菜单栏5 鼠标 一 什么是Gazebo x
  • Mac Pro安装Python3

    用brew install python3报错 xff1a curl 7 Failed to connect to raw githubusercontent com port 443 Operation 问题解决 xff1a https
  • 基于宽度优先爬虫

    coding utf 8 from selenium import webdriver import time from bs4 import BeautifulSoup import urllib2 from selenium webdr
  • LINUX amixer工具及kcontrol使用

    网上对amixer工具的使用较多 xff0c 但使用kcontrol coding的例子有限 xff0c 目前比较好的一篇文章如下 xff1a http blog chinaunix net uid 27875 id 5783294 htm
  • 小波部分特征说明和小波特征比较---未写完

    小波特征 1 xff1a 多分辨分析可以对信号进行有效的时频分分解 xff0c 但是由于其尺度函数是按二进制变化的 xff0c 因此在高频段器频率分辨率较差 xff0c 对信号的频段进行指数等间隔划分 小波包能够为信号提供一种更加精细的分析
  • 《国产嵌入式微系统msOS成型记》之终结篇——嵌入式微系统msOS的出发点:应用

    http www eefocus com embedded 329047 msOS设计的本意是为电子类设备及PLC设计的 xff0c 一开始想着仿造西门子的S7 300来设计一套基于CAN通讯的分布式PLC xff0c 然而当设计了几个硬件
  • Debian10(英文环境)安装中文输入法fcitx

    执行 locale 以及 cat etc locale gen 查看当前数据 sudo dpkg reconfigure locales 本身默认已经选择了 en US UTF 8 UTF 8 增加几个zh CN选项 然后ok 然后这里继续
  • 小批量数据测试模型容易出现Bug

    如果是直接用的原来数据集类的数据 xff0c 截取出来后 xff0c 数据范围为 0 255 xff0c 需要手动实现transform的功能
  • Windbg介绍

    文章目录 1 介绍1 1 相关网站1 2 下载1 3 符号配置 2 常用命令2 1 常用的标准命令2 2 常用的 96 96 命令有2 3 常用的 96 96 命令有 3 调试环境搭建3 1 实时调试进程3 2 调试进程dump3 3 调试

随机推荐

  • 笔记本电脑 (天梯图) 处理器 和 显卡

    游戏本一般是i5 9300H起步 xff0c 低于这个标准的游戏本不建议购买 另外需要注意的是不要被10代CPU唬住了 xff0c 拿最近卖的比较火的i5 10210U来说 xff0c 其性能还不如8代的i5 8300H 我们以移动版GTX
  • ActiveX开发详解

    配套源码 http download csdn net detail mingojiang 4743623 转载请注明出自 http blog csdn net MingoJiang 一 ActiveX基础 1 1什么是ActiveX Ac
  • GNVM - Node.js 多版本管理器

    GNVM Node js 多版本管理器 GNVM 是一个简单的 Windows 下 Node js 多版本管理器 xff0c 类似的 nvm nvmw nodist c gt gnvm install latest 1 0 0 x86 1
  • hadoop入门-wordcount

    hadoop是阿帕奇基金会的一个顶级项目 xff0c 主要用于大量的廉价机器组成的集群去执行大规模运算 xff0c 主要是海量数据的处理 在hadoop官网 xff08 http hadoop apache org xff09 hadoop
  • aptitude和apt-get的区别和联系【转,有添加和修改】

    起初GNU Linux系统中只有 tar gz 用户 必须自己编译他们想使用的每一个程序 在Debian出现之後 xff0c 人们认为有必要在系统 中添加一种机 制用来管理 安装在计算机上的软件包 人们将这套系统称为dpkg 至此着名的 p
  • Mac终端配置代理

    export http proxy 61 socks5 127 0 0 1 49719 配置http访问 export https proxy 61 socks5 127 0 0 1 49719 配置https export all pro
  • 如何将Java程序打成可执行jar包

    前几天 xff0c 公司运维找我让我帮他写个Java小程序 xff0c 读取磁盘指定目录的文件 xff0c 然后根据读取的内容查询第三方接口 xff0c 再将第三方接口响应的数据写入磁盘文件 然后我花了半天给他写了这个小程序 xff0c 但
  • javafx_scenebuilder-2_0-windows.msi 百度云盘下载

    javafx scene builder 官网下载很慢 网上有很多人分享 xff0c 都要付积分下载 下面是从官网下载好的 xff0c 传我百度网盘了 xff0c 有需要的大家去下载吧 链接 xff1a https pan baidu co
  • 100+套Axure数据可视化大屏展示原型模板及通用主键库

    内置多种实用美观的可视化组件库及行业模板库 xff0c 行业模板涵盖 xff1a 金融 教育 医疗 政府 交通 制造等多个行业 xff0c 提供设计参考 随着大数据的发展 xff0c 可视化大屏在各行各业得到越来越广泛的应用 可视化大屏不再
  • 数据可视化大屏UI界面

    数据可视化大屏 科技大屏展示 智慧城市 智慧农业 领导页展示大屏 PSD文件 UI可视化大屏模板PSD文件 156套可视化大屏PSD设计文件 xff0c 送给有需要的人 格式格式 xff1a PSD jpg 适合人群 xff1a 可视化大屏
  • SpringBoot 自定义注解实现Redis缓存功能

    背景 最近小A的公司要做一个大屏可视化平台 xff0c 主要是给领导看的 xff0c 领导说这个项目要给领导演示 xff0c 效果好不好直接关系到能不能拿下这个项目 xff0c 领导还补了一句 这项目至少是百万级的 xff0c 大伙要全力以
  • Linux 下 chmod 777 修改权限

    一 rwxrwxrwx 777 Unix Linux 的操作系统 xff0c 每个文件 文件夹也被看作是文件 都按读 写 运行设定权限 例如用ls l命令列文件表时 xff0c 得到如下输出 xff1a rw r r 1 mchopin u
  • Spring创建对象初始化bean的时机分为两种形式:

    import org junit Test import org springframework context ApplicationContext import org springframework context support C
  • 页面动态数据的滚动效果——jquery滚动组件(vticker.js)

    lt script language 61 34 javascript 34 src 61 34 lirms Test jquery 1 4 2 js 34 gt lt script gt lt script language 61 34
  • MySQL 查询结果以百分比显示

    找了一些资料 xff0c 然后我是用到了MySQL字符串处理中的两个函数concat 和left 1 span style color ff0000 CONCAT span str1 str2 返回来自于参数连结的字符串 如果任何参数是 N
  • rpm包安装过程中依赖问题“libc.so.6 is needed by XXX”解决方法

    转自 xff1a http raksmart idcspy com 781 rpm包安装过程中依赖问题 libc so 6 is needed by XXX 解决方法 与本教程高度相关文章 xff08 读完应该可以解决你的问题 xff09
  • 艺博: linux命令大宝典系列之mkdir创建目录

    在成长的过程中 人总要经历一些痛苦和挫折才能更加成熟与坚强 目录 mkdir是什么mkdir的语法mkdir的选项含义mkdir的实例1 使用mkdir命令创建一个dir1目录 默认权限775 2 使用mkdir m命令新建一个dir2目录
  • linux命令xrandr修改桌面分辨率

    xrandr临时修改分辨率 方法一 打开终端xrandr Screen 0 minimum 8 x 8 current 1366 x 768 maximum 32767 x 32767 eDP1 connected primary 1366
  • Python 教你训练一个98%准确率的微博抑郁文本分类模型(含数据)

    Paddle是一个比较高级的深度学习开发框架 xff0c 其内置了许多方便的计算单元可供使用 xff0c 我们之前写过PaddleHub相关的文章 xff1a 1 Python 识别文本情感就这么简单 2 比PS还好用 xff01 Pyth
  • 记一次神奇的时间转换问题(SheetJS)

    最近在写一个功能 xff0c 使用SheetJS读取Excel表格 xff0c 在读取日期的时候发现了一个隐藏很深的坑 xff0c 特此记录一下 SheetJS读取Excel文件时 xff0c 可指定参数 cellDates true xf