欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

js完美地处理转换 Excel 的日期格式

程序员文章站 2022-03-18 19:48:15
...

问题提出

在使用 js 做 Excel 文件读取时,我们发现对于日期的处理很棘手。主要存在以下几种情况:

  • 所有的日期单元格读取到的是一个类似 44211.7533 的数字的问题
  • 格式不规范问题,如 2021-1-1 0:0:00
  • 按指定格式格式化的问题

带着这几个问题,我在网上找了几篇文章,但是都不能很好地解决,所以自己动手解决,并整理分享出来,希望对大家有所帮助

解决方案

/**
 * 处理从 Excel 单元格中读取到的日期格式的字段并转换为指定格式的日期字符串
 * 
 * @param {string|number} date 从 Excel 单元格中读取到的日期格式
 * @param {string} fmt 要转换为的格式,默认 yyy-MM-dd HH:mm:ss
 */
function handleDate(date, fmt) {
  if (!date) {
    return date;
  }
  let dateNum = parseFloat(date);
  if (dateNum > 0) {
    // 大于 1000 万说明是一个毫秒数,直接解析并转换为指定格式即可
    if (dateNum > 10000000) {
      return formatDate(dateNum, fmt);
    }
    // 否则认为这个是一个 Excel 格式的日期
    date = formatExcelDate(dateNum, fmt);
  }
  // 将不规则的格式,例如 "2020-1-1     1:3:3" 转换成 yyyy-MM-dd HH:mm:ss
  // 再转为 Date 对象进行指定的格式化
  console.log(date, typeof (date))
  let dtPars = date.split(/\s+/g)
  let dPars = dtPars[0].split("-");
  dPars[1] = padding2(dPars[1])
  dPars[2] = padding2(dPars[2])
  dtPars[0] = dPars.join("-");
  let tPars = dtPars[1].split(":");
  tPars[0] = padding2(tPars[0])
  // 支持分缺失
  tPars[1] = padding2(tPars[1] || "00")
  // 支持秒缺失
  tPars[2] = padding2(tPars[2] || "00")
  dtPars[1] = tPars.join(":");
  return formatDate(new Date(dtPars.join(" ")), fmt);
}

/**
 * 缺位补0
 */
function padding2(part) {
  if (part.length === 1) {
    return "0" + part
  } else {
    return part;
  }
}

/**
 * 解析Excel表达的日期数字,并转换为指定格式的日期字符串
 * 
 * @param {number} numb Excel解析出的数字形式的日期
 * @param {string} format 要转换为的格式,默认 yyy-MM-dd HH:mm:ss
 */
function formatExcelDate(numb, format) {
  const time = new Date((numb - 2) * 24 * 3600000 + 1);
  time.setYear(time.getFullYear() - 70);
  time.setHours(time.getHours() - 8);
  return formatDate(time, format);
}

/**
 * 日期格式转换
 * 
 * @param {date|number} date 日期
 * @param {string} fmt 要转换为的格式,默认 yyy-MM-dd HH:mm:ss
 */
function formatDate(date, fmt) {
  date = date == undefined ? new Date() : date;
  date = typeof date == 'number' ? new Date(date) : date;
  fmt = fmt || 'yyyy-MM-dd HH:mm:ss';
  let obj =
  {
    'y': date.getFullYear(), // 年份,注意必须用getFullYear
    'M': date.getMonth() + 1, // 月份,注意是从0-11
    'd': date.getDate(), // 日期
    'q': Math.floor((date.getMonth() + 3) / 3), // 季度
    'w': date.getDay(), // 星期,注意是0-6
    'H': date.getHours(), // 24小时制
    'h': date.getHours() % 12 == 0 ? 12 : date.getHours() % 12, // 12小时制
    'm': date.getMinutes(), // 分钟
    's': date.getSeconds(), // 秒
    'S': date.getMilliseconds() // 毫秒
  };
  let week = ['天', '一', '二', '三', '四', '五', '六'];
  for (let i in obj) {
    fmt = fmt.replace(new RegExp(i + '+', 'g'), function (m) {
      let val = obj[i] + '';
      if (i == 'w') return (m.length > 2 ? '星期' : '周') + week[val];
      for (let j = 0, len = val.length; j < m.length - len; j++) val = '0' + val;
      return m.length == 1 ? val : val.substring(val.length - m.length);
    });
  }
  return fmt;
}

使用示例

假设我们有这样一张 Excel 表:

id 姓名 签到时间 签退时间
1 张三 2021/1/1 8:00:00 2021/1/1 18:00:00
2 李四 2021/1/1 9:30:00 2021/1/1 19:15:00

我们使用 node-xlsx 读取:

const nodeXlsx = require('node-xlsx')	//引用node-xlsx模块
var fs = require('fs');
// 读取excel表格
const ex1 = nodeXlsx.parse(filePath)

//取出excel文件中的第一个工作表中的全部数据
let excel_content = ex1[0].data

// 删掉表头
excel_content.splice(0, 1);
let records = excel_content.map(row => {
  return {
    id: row[0],
    name: row[1],
    startTime: handleDate(row[2], 'yyyy-MM-dd HH:mm:ss'),
    endTime: handleDate(row[3], 'yyyy-MM-dd HH:mm:ss')
  }
});
console.log(JSON.stringify(records))

这时我们将得到正确的日期格式结果:

[{"id": 1, "name": "张三", "startTime": "2021-01-01 08:00:00", "endTime": "2021-01-01 18:00:00"},
{"id": 2, "name": "李四", "startTime": "2021-01-01 09:30:00", "endTime": "2021-01-01 19:15:00"}]