我想在JavaScript中解析一个没有时区的日期。我试着:

new Date(Date.parse("2005-07-08T00:00:00+0000"));

返回2005年7月8日星期五02:00:00 GMT+0200(中欧夏令时):

new Date(Date.parse("2005-07-08 00:00:00 GMT+0000"));

返回相同的结果,并且:

new Date(Date.parse("2005-07-08 00:00:00 GMT-0000"));

也返回相同的结果。

我想解析时间:

没有时区。 而不调用构造函数Date。UTC或new日期(年、月、日)。 通过简单地将一个字符串传递到Date构造函数(没有原型方法)。

我必须生成一个Date对象,而不是String对象。


当前回答

因为它确实是一个格式问题时显示日期(例如显示在本地时间),我喜欢使用新的(ish) Intl。DateTimeFormat对象来执行格式化,因为它更显式,并提供更多输出选项:

const dateOptions = { timeZone: 'UTC', month: 'long', day: 'numeric', year: 'numeric' };

const dateFormatter = new Intl.DateTimeFormat('en-US', dateOptions);
const dateAsFormattedString = dateFormatter.format(new Date('2019-06-01T00:00:00.000+00:00'));

console.log(dateAsFormattedString) // "June 1, 2019"

如所示,通过将timeZone设置为“UTC”,它将不会执行本地转换。作为奖励,它还允许您创建更精致的输出。你可以阅读更多关于国际。DateTimeFormat对象。

无需创建新的Intl就可以实现相同的功能。DateTimeFormat对象。简单地将语言环境和日期选项直接传递给toLocaleDateString()函数。

const dateOptions = { timeZone: 'UTC', month: 'long', day: 'numeric', year: 'numeric' };
const myDate = new Date('2019-06-01T00:00:00.000+00:00');
myDate.toLocaleDateString('en-US', dateOptions); // "June 1, 2019"

其他回答

这里有一个简单的解决方案:

const handler1 = {
  construct(target, args) {
    let newDate = new target(...args);
    var tzDifference = newDate.getTimezoneOffset();
    return new target(newDate.getTime() + tzDifference * 60 * 1000);
  }
};

Date = new Proxy(Date, handler1);

Date对象本身无论如何都将包含timezone,返回的结果是以默认方式将其转换为字符串的结果。也就是说,你不能创建一个没有时区的日期对象。但是您可以通过创建自己的Date对象来模仿Date对象的行为。 然而,这最好交给像moment.js这样的库。

因为它确实是一个格式问题时显示日期(例如显示在本地时间),我喜欢使用新的(ish) Intl。DateTimeFormat对象来执行格式化,因为它更显式,并提供更多输出选项:

const dateOptions = { timeZone: 'UTC', month: 'long', day: 'numeric', year: 'numeric' };

const dateFormatter = new Intl.DateTimeFormat('en-US', dateOptions);
const dateAsFormattedString = dateFormatter.format(new Date('2019-06-01T00:00:00.000+00:00'));

console.log(dateAsFormattedString) // "June 1, 2019"

如所示,通过将timeZone设置为“UTC”,它将不会执行本地转换。作为奖励,它还允许您创建更精致的输出。你可以阅读更多关于国际。DateTimeFormat对象。

无需创建新的Intl就可以实现相同的功能。DateTimeFormat对象。简单地将语言环境和日期选项直接传递给toLocaleDateString()函数。

const dateOptions = { timeZone: 'UTC', month: 'long', day: 'numeric', year: 'numeric' };
const myDate = new Date('2019-06-01T00:00:00.000+00:00');
myDate.toLocaleDateString('en-US', dateOptions); // "June 1, 2019"

我遇到了同样的问题,然后想起了我正在从事的一个遗留项目的一些问题,以及他们是如何处理这个问题的。当时我并不理解,也并不在意,直到我自己遇到了这个问题

var date = '2014-01-02T00:00:00.000Z'
date = date.substring(0,10).split('-')
date = date[1] + '-' + date[2] + '-' + date[0]

new Date(date) #Thu Jan 02 2014 00:00:00 GMT-0600

不管出于什么原因,将日期传递为“01-02-2014”将时区设置为0并忽略用户的时区。这可能是Date类中的一个侥幸,但它在一段时间以前就存在,今天也存在。而且它似乎可以跨浏览器工作。你自己试试吧。

这段代码是在一个全局项目中实现的,在这个项目中,时区很重要,但查看日期的人并不关心它被引入的确切时间。

JavaScript中的日期只是在内部保持简单,因此日期-时间数据以UTC Unix历元(毫秒或毫秒)存储。

如果你想有一个“固定”的时间,无论你在地球上的任何时区都不会改变,你可以调整UTC时间以匹配当前的本地时区并保存它。在检索它时,无论您所在的本地时区是什么,它都会根据保存它的时区显示调整后的UTC时间,然后添加本地时区偏移量以获得“固定”时间。

保存日期(毫秒):

toUTC(datetime) {
  const myDate = (typeof datetime === 'number')
    ? new Date(datetime)
    : datetime;

  if (!myDate || (typeof myDate.getTime !== 'function')) {
    return 0;
  }

  const getUTC = myDate.getTime();
  const offset = myDate.getTimezoneOffset() * 60000; // It's in minutes so convert to ms
  return getUTC - offset; // UTC - OFFSET
}

检索/显示日期(毫秒):

fromUTC(datetime) {
  const myDate = (typeof datetime === 'number')
    ? new Date(datetime)
    : datetime;

  if (!myDate || (typeof myDate.getTime !== 'function')) {
    return 0;
  }

  const getUTC = myDate.getTime();
  const offset = myDate.getTimezoneOffset() * 60000; // It's in minutes so convert to ms
  return getUTC + offset; // UTC + OFFSET
}

然后你可以:

const saveTime = new Date(toUTC(Date.parse("2005-07-08T00:00:00+0000")));
// SEND TO DB....

// FROM DB...
const showTime = new Date(fromUTC(saveTime));