我有一个网页,有三个下拉菜单,分别是日、月和年。如果我使用JavaScript Date构造函数接受数字,那么我得到一个Date对象为我的当前时区:

new Date(xiYear, xiMonth, xiDate)

给出正确的日期,但由于夏令时,它认为日期是GMT+01:00。

这里的问题是,然后我将这个日期传递给一个Ajax方法,当日期在服务器上被反序列化时,它已经转换为GMT,因此失去了一个小时,这将一天往回移动了一个小时。 现在我可以将日、月和年分别传递到Ajax方法中,但似乎应该有更好的方法。

接受的答案为我指明了正确的方向,但是仅仅使用setUTCHours()本身就改变了:

Apr 5th 00:00 GMT+01:00 

to

Apr 4th 23:00 GMT+01:00

然后,我还必须设置UTC日期、月份和年份

Apr 5th 01:00 GMT+01:00

这正是我想要的


当前回答

单线解决方案

new Date(new Date(1422524805305).getTime() - 330*60*1000)

使用以毫秒为单位的时间戳,而不是1422524805305 不要使用330,而是使用您的时区偏移量(分钟wrt)。格林尼治时间(例如印度+5:30是5*60+30 = 330分钟)

其他回答

我脑子疼。 如何获得当前相同的日期与时区,如在服务器与WORDPRESS到js。

var dnow = new Date();
dnow = dnow.getTime() + dnow.getTimezoneOffset()*60*1000 + <?= get_option( 'gmt_offset' ) ?> * 60*60*1000;

首先,我们需要得到当前的日期。然后我们需要删除用户时区偏移。然后我们从WordPress选项页面添加gmt偏移量。

我使用了timezone-js包。

var timezoneJS  = require('timezone-js');
var tzdata = require('tzdata');

createDate(dateObj) {
    if ( dateObj == null ) {
        return null;
    }
    var nativeTimezoneOffset = new Date().getTimezoneOffset();
    var offset = this.getTimeZoneOffset();

    // use the native Date object if the timezone matches
    if ( offset == -1 * nativeTimezoneOffset ) {
        return dateObj;
    }

    this.loadTimeZones();

    // FIXME: it would be better if timezoneJS.Date was an instanceof of Date
    //        tried jquery $.extend
    //        added hack to Fiterpickr to look for Dater.getTime instead of "d instanceof Date"
    return new timezoneJS.Date(dateObj,this.getTimeZoneName());
},
var d = new Date(xiYear, xiMonth, xiDate);
d.setTime( d.getTime() + d.getTimezoneOffset()*60*1000 );

这个答案是专门为最初的问题量身定制的,不会给出你所期望的答案。特别是,有些人想要减去时区偏移量,而不是加上它。记住,这个解决方案的重点是针对特定的反序列化破解javascript的date对象,而不是在所有情况下都正确。

当我创建一个日期对象:

日期(年、月、日、时、分)

我在本地主机上工作得很好。 当我部署到服务器时,它崩溃了,因为服务器在另一个时区。

我不能使用getTimezoneOffset()。我需要我的家的时区偏移-取决于夏季/冬季时间

// add diff minutes between myself (HOME) and server 
timezoneHomeOffset (d, tz = 'Europe/Copenhagen') {
  const utc = new Date(d.getTime())
  const dHome = new Date(d.toLocaleString('en-US', { timeZone: tz }))
  const diff = Math.round((utc - dHome) / 60000) // 60*1000 => minutes
  d.setMinutes(d.getMinutes() + diff)
  return d
}

其实做起来一点也不难,但肯定不是凭直觉就能得出解决方案的。这里有一些非常复杂的答案(尽管也有一些不错的答案)。下面是我为确保我的服务器时间戳与我的本地时间戳匹配而提出的方法,无论我部署的服务器恰好位于哪个时区。

(CET =中欧时区,正好是我个人的时区;你可以得到任何给定时区的偏移量并计算它,如果你愿意,甚至可以把它作为一个参数,但对于我的目的,我只需要让我的日期都是所需的一致时区。)

    const convertDateToCET = function(date) {
        date = new Date(date)
        // let startTime = date.getTime();
        const cetOffset = -120; // this is the number you get from running 
        // `(new Date()).getTimezoneOffset()` if you're on a machine in CET
        const offsetFromCET = (date.getTimezoneOffset() - cetOffset);
        const cetMillsecondOffset = ( cetOffset* 60 * 1000);
        date = new Date( date.getTime() - cetMillsecondOffset ) 
        // let endTime = date.getTime()
        // console.log("updated date from",startTime,"to",endTime)
        return date;
    },

用这个短语,你可以按照你所期望的方式来定一个时间。

    let myDate = new Date("12-4-2021")
    myDate.setHour(14)
    myDate.setMinute(30)
    // now myDate is 2:30pm, December 4th, 2021, in whatever the timezone the machine of code running happens to be in
    myDate = convertDateToCET(myDate)
    // now myDate will show up as 2:30pm, Dec 4th, 2021, mapped into your local timezone
    // so, if you're in the UK, and one hour behind CET, myDate is now 1:30pm, Dec 4th, 2021

这里的关键是date.getTimezoneOffset()。如果你参加了四六级考试,那么这个数字就是-120,这样就消去了,没有区别(所以四六级考试的结果就是四六级考试的结果)。如果你在英国,比CET晚一小时,那么输出将是-60,这意味着-60 + 120 = +60,这导致我们将输入时间改变一小时,以此类推。

对于这种情况,转换所有内容并使用UTC可能更有意义,但考虑到我的所有输入时间都是使用CET,而且我最初是根据机器上的本地情况开发系统的,因此这个实用程序允许我通过在几个关键位置调用这个函数来转换现有代码。

注意:请确保不要在同一日期多次应用此函数调用,因为您将多次重新应用偏移量,将其抛出!