使用设置的时区创建日期而不使用字符串表示形式


412

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

new Date(xiYear, xiMonth, xiDate)

输入正确的日期,但由于夏令时,它认为该日期为GMT + 01:00。

这里的问题是,我然后将其传递Date给Ajax方法,并且在服务器上反序列化日期时,日期已转换为GMT,因此浪费了一个小时,使日期倒退了一个小时。现在,我可以将日期,月份和年份分别传递给Ajax方法,但是似乎应该有更好的方法。

公认的答案为我指明了正确的方向,但是仅仅使用setUTCHours()它本身就发生了变化:

Apr 5th 00:00 GMT+01:00 

Apr 4th 23:00 GMT+01:00

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

Apr 5th 01:00 GMT+01:00

这就是我想要的。


9
如果已接受的答案将您引向正确的方向,但没有回答您的问题,那么我认为这不应是已接受的答案。答案应该回答所问的问题。
TWR Cole

Answers:


481

使用.setUTCHours()它可以实际以UTC时间设置日期,这将使您可以在整个系统中使用UTC时间。

但是,除非指定日期字符串,否则不能在构造函数中使用UTC进行设置。

使用new Date(Date.UTC(year, month, day, hour, minute, second))可以从特定UTC时间创建日期对象。


101
“ new Date(Date.UTC(...))”语法使您可以创建一个日期,该日期在它表示的时间点上等同于UTC日期,但并不相同-它具有一个不同(非UTC)时区。
安东尼

52
请记住,使用“日期”时,“月”值的范围是0-11(而不是1-12)。我一直得到2h的时区偏移(应该是1h),花了我几个小时才发现原因是错误的月份。
Select0r

4
这个答案很好。但是我正在使用在很多地方都使用新日期的库[datepicker ui]。我想要的只是设置UTC时区,每个日期都与新时区相同。我很惊讶Javascript对此一无所有。
Sanjeev Kumar Dangi 2012年

6
@jishi-日期对象基于UTC时间值,而不是本地时间。但是,默认的Date.prototype.toString方法将显示本地时间值。
RobG

5
@Anthony —“ 但不是同一时间 ”是不正确的。它表示时间完全相同,唯一的区别是时区偏移。
RobG

198
var d = new Date(xiYear, xiMonth, xiDate);
d.setTime( d.getTime() + d.getTimezoneOffset()*60*1000 );

该答案是专门为原始问题量身定做的,不会提供您必然期望的答案。特别是,有些人会希望减去时区偏移而不是添加。请记住,尽管此解决方案的重点是针对特定的反序列化破解javascript的date对象,但并非在所有情况下都是正确的。


62
@gthmb当然可以,但是我觉得*60*1000在这种情况下更清楚了;换句话说,它在那里的原因是不言而喻的。
TWR Cole

22
这对我几乎是有效的,除了我必须使用-(减号)而不是+(加号)来获取我所在时区的正确时间。
Wytze 2014年

3
是的,正如其他人指出的那样-我认为此答案有误。应该是负号而不是正号。
UpTheCreek 2014年

3
根据developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/…的定义,getTimezoneOffset返回的值是根据调用函数时语言环境中的实际偏移量进行签名的,其​​中包括DST,所以我不明白为什么你需要减去它。
TWR Cole

15
如果将timezoneOffset添加到date对象,则其在本地时区中格式设置的值将看起来像UTC中的正确值,但仍具有原始时区偏移量(某些表示形式,例如“ ISOString”将实际显示出来)。因此,取决于您随后如何序列化日期对象,JS可能会再次应用时区偏移,从而给您错误的答案。我认为这是造成+/-之间这些注释混乱的原因。无论如何,我的反对意见是出于这个事实,也是出于“在大多数情况下您会得到期望的结果”。
metamatt

173

我相信您需要createDateAsUTC函数(请与convertDateToUTC比较)

function createDateAsUTC(date) {
    return new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate(), date.getHours(), date.getMinutes(), date.getSeconds()));
}

function convertDateToUTC(date) { 
    return new Date(date.getUTCFullYear(), date.getUTCMonth(), date.getUTCDate(), date.getUTCHours(), date.getUTCMinutes(), date.getUTCSeconds()); 
}

36
我为他的回答清晰和有用而感到惊讶。直到今天才知道使用Javascript日期是一场噩梦:S
will824 2014年

介意解释两者之间的区别?第一个转换date为UTC时区,但是第二个似乎没有任何用处?(返回的日期与相同date
Jonathan Lin

4
我现在明白了:第一个返回UTC时区中的日期,以及本地时间的文字日期值。第二个返回本地时区中的日期,但带有UTC文字日期值。
林纳森(Jonathan Lin)

8
此方法是一种称为“历元移位”的模式的实现,该模式旨在将历元(基于UTC)移动到以当前时区偏移量偏移的历元。不幸的是,即使这种情况很普遍,但这种方法还是有缺陷的。JavaScript的Date对象将始终反映基于UTC的UNIX纪元和本地时区。当您调用toString结果日期对象并仍然看到本地时区时,症状很明显,即使您希望它位于UTC时也是如此。
马特·约翰逊·品脱

2
在当地时区的夏令时转换附近,它也可能导致时间值错误。简而言之,时代转移(通过任何实现)不适用于JavaScript Date对象。在这里看到这种情况的另一种方法是Date.UTC期望基于UTC的值,并且您将其提供给本地时间值,反之亦然Date
马特·约翰逊·品脱

70

只需设置时区并取回

new Date().toLocaleString("en-US", {timeZone: "America/New_York"})

其他时区如下

var world_timezones =
[
    'Europe/Andorra',
    'Asia/Dubai',
    'Asia/Kabul',
    'Europe/Tirane',
    'Asia/Yerevan',
    'Antarctica/Casey',
    'Antarctica/Davis',
    'Antarctica/DumontDUrville', 
    'Antarctica/Mawson',
    'Antarctica/Palmer',
    'Antarctica/Rothera',
    'Antarctica/Syowa',
    'Antarctica/Troll',
    'Antarctica/Vostok',
    'America/Argentina/Buenos_Aires',
    'America/Argentina/Cordoba',
    'America/Argentina/Salta',
    'America/Argentina/Jujuy',
    'America/Argentina/Tucuman',
    'America/Argentina/Catamarca',
    'America/Argentina/La_Rioja',
    'America/Argentina/San_Juan',
    'America/Argentina/Mendoza',
    'America/Argentina/San_Luis',
    'America/Argentina/Rio_Gallegos',
    'America/Argentina/Ushuaia',
    'Pacific/Pago_Pago',
    'Europe/Vienna',
    'Australia/Lord_Howe',
    'Antarctica/Macquarie',
    'Australia/Hobart',
    'Australia/Currie',
    'Australia/Melbourne',
    'Australia/Sydney',
    'Australia/Broken_Hill',
    'Australia/Brisbane',
    'Australia/Lindeman',
    'Australia/Adelaide',
    'Australia/Darwin',
    'Australia/Perth',
    'Australia/Eucla',
    'Asia/Baku',
    'America/Barbados',
    'Asia/Dhaka',
    'Europe/Brussels',
    'Europe/Sofia',
    'Atlantic/Bermuda',
    'Asia/Brunei',
    'America/La_Paz',
    'America/Noronha',
    'America/Belem',
    'America/Fortaleza',
    'America/Recife',
    'America/Araguaina',
    'America/Maceio',
    'America/Bahia',
    'America/Sao_Paulo',
    'America/Campo_Grande',
    'America/Cuiaba',
    'America/Santarem',
    'America/Porto_Velho',
    'America/Boa_Vista',
    'America/Manaus',
    'America/Eirunepe',
    'America/Rio_Branco',
    'America/Nassau',
    'Asia/Thimphu',
    'Europe/Minsk',
    'America/Belize',
    'America/St_Johns',
    'America/Halifax',
    'America/Glace_Bay',
    'America/Moncton',
    'America/Goose_Bay',
    'America/Blanc-Sablon',
    'America/Toronto',
    'America/Nipigon',
    'America/Thunder_Bay',
    'America/Iqaluit',
    'America/Pangnirtung',
    'America/Atikokan',
    'America/Winnipeg',
    'America/Rainy_River',
    'America/Resolute',
    'America/Rankin_Inlet',
    'America/Regina',
    'America/Swift_Current',
    'America/Edmonton',
    'America/Cambridge_Bay',
    'America/Yellowknife',
    'America/Inuvik',
    'America/Creston',
    'America/Dawson_Creek',
    'America/Fort_Nelson',
    'America/Vancouver',
    'America/Whitehorse',
    'America/Dawson',
    'Indian/Cocos',
    'Europe/Zurich',
    'Africa/Abidjan',
    'Pacific/Rarotonga',
    'America/Santiago',
    'America/Punta_Arenas',
    'Pacific/Easter',
    'Asia/Shanghai',
    'Asia/Urumqi',
    'America/Bogota',
    'America/Costa_Rica',
    'America/Havana',
    'Atlantic/Cape_Verde',
    'America/Curacao',
    'Indian/Christmas',
    'Asia/Nicosia',
    'Asia/Famagusta',
    'Europe/Prague',
    'Europe/Berlin',
    'Europe/Copenhagen',
    'America/Santo_Domingo',
    'Africa/Algiers',
    'America/Guayaquil',
    'Pacific/Galapagos',
    'Europe/Tallinn',
    'Africa/Cairo',
    'Africa/El_Aaiun',
    'Europe/Madrid',
    'Africa/Ceuta',
    'Atlantic/Canary',
    'Europe/Helsinki',
    'Pacific/Fiji',
    'Atlantic/Stanley',
    'Pacific/Chuuk',
    'Pacific/Pohnpei',
    'Pacific/Kosrae',
    'Atlantic/Faroe',
    'Europe/Paris',
    'Europe/London',
    'Asia/Tbilisi',
    'America/Cayenne',
    'Africa/Accra',
    'Europe/Gibraltar',
    'America/Godthab',
    'America/Danmarkshavn',
    'America/Scoresbysund',
    'America/Thule',
    'Europe/Athens',
    'Atlantic/South_Georgia',
    'America/Guatemala',
    'Pacific/Guam',
    'Africa/Bissau',
    'America/Guyana',
    'Asia/Hong_Kong',
    'America/Tegucigalpa',
    'America/Port-au-Prince',
    'Europe/Budapest',
    'Asia/Jakarta',
    'Asia/Pontianak',
    'Asia/Makassar',
    'Asia/Jayapura',
    'Europe/Dublin',
    'Asia/Jerusalem',
    'Asia/Kolkata',
    'Indian/Chagos',
    'Asia/Baghdad',
    'Asia/Tehran',
    'Atlantic/Reykjavik',
    'Europe/Rome',
    'America/Jamaica',
    'Asia/Amman',
    'Asia/Tokyo',
    'Africa/Nairobi',
    'Asia/Bishkek',
    'Pacific/Tarawa',
    'Pacific/Enderbury',
    'Pacific/Kiritimati',
    'Asia/Pyongyang',
    'Asia/Seoul',
    'Asia/Almaty',
    'Asia/Qyzylorda',
    'Asia/Qostanay', 
    'Asia/Aqtobe',
    'Asia/Aqtau',
    'Asia/Atyrau',
    'Asia/Oral',
    'Asia/Beirut',
    'Asia/Colombo',
    'Africa/Monrovia',
    'Europe/Vilnius',
    'Europe/Luxembourg',
    'Europe/Riga',
    'Africa/Tripoli',
    'Africa/Casablanca',
    'Europe/Monaco',
    'Europe/Chisinau',
    'Pacific/Majuro',
    'Pacific/Kwajalein',
    'Asia/Yangon',
    'Asia/Ulaanbaatar',
    'Asia/Hovd',
    'Asia/Choibalsan',
    'Asia/Macau',
    'America/Martinique',
    'Europe/Malta',
    'Indian/Mauritius',
    'Indian/Maldives',
    'America/Mexico_City',
    'America/Cancun',
    'America/Merida',
    'America/Monterrey',
    'America/Matamoros',
    'America/Mazatlan',
    'America/Chihuahua',
    'America/Ojinaga',
    'America/Hermosillo',
    'America/Tijuana',
    'America/Bahia_Banderas',
    'Asia/Kuala_Lumpur',
    'Asia/Kuching',
    'Africa/Maputo',
    'Africa/Windhoek',
    'Pacific/Noumea',
    'Pacific/Norfolk',
    'Africa/Lagos',
    'America/Managua',
    'Europe/Amsterdam',
    'Europe/Oslo',
    'Asia/Kathmandu',
    'Pacific/Nauru',
    'Pacific/Niue',
    'Pacific/Auckland',
    'Pacific/Chatham',
    'America/Panama',
    'America/Lima',
    'Pacific/Tahiti',
    'Pacific/Marquesas',
    'Pacific/Gambier',
    'Pacific/Port_Moresby',
    'Pacific/Bougainville',
    'Asia/Manila',
    'Asia/Karachi',
    'Europe/Warsaw',
    'America/Miquelon',
    'Pacific/Pitcairn',
    'America/Puerto_Rico',
    'Asia/Gaza',
    'Asia/Hebron',
    'Europe/Lisbon',
    'Atlantic/Madeira',
    'Atlantic/Azores',
    'Pacific/Palau',
    'America/Asuncion',
    'Asia/Qatar',
    'Indian/Reunion',
    'Europe/Bucharest',
    'Europe/Belgrade',
    'Europe/Kaliningrad',
    'Europe/Moscow',
    'Europe/Simferopol',
    'Europe/Kirov',
    'Europe/Astrakhan',
    'Europe/Volgograd',
    'Europe/Saratov',
    'Europe/Ulyanovsk',
    'Europe/Samara',
    'Asia/Yekaterinburg',
    'Asia/Omsk',
    'Asia/Novosibirsk',
    'Asia/Barnaul',
    'Asia/Tomsk',
    'Asia/Novokuznetsk',
    'Asia/Krasnoyarsk',
    'Asia/Irkutsk',
    'Asia/Chita',
    'Asia/Yakutsk',
    'Asia/Khandyga',
    'Asia/Vladivostok',
    'Asia/Ust-Nera',
    'Asia/Magadan',
    'Asia/Sakhalin',
    'Asia/Srednekolymsk',
    'Asia/Kamchatka',
    'Asia/Anadyr',
    'Asia/Riyadh',
    'Pacific/Guadalcanal',
    'Indian/Mahe',
    'Africa/Khartoum',
    'Europe/Stockholm',
    'Asia/Singapore',
    'America/Paramaribo',
    'Africa/Juba',
    'Africa/Sao_Tome',
    'America/El_Salvador',
    'Asia/Damascus',
    'America/Grand_Turk',
    'Africa/Ndjamena',
    'Indian/Kerguelen',
    'Asia/Bangkok',
    'Asia/Dushanbe',
    'Pacific/Fakaofo',
    'Asia/Dili',
    'Asia/Ashgabat',
    'Africa/Tunis',
    'Pacific/Tongatapu',
    'Europe/Istanbul',
    'America/Port_of_Spain',
    'Pacific/Funafuti',
    'Asia/Taipei',
    'Europe/Kiev',
    'Europe/Uzhgorod',
    'Europe/Zaporozhye',
    'Pacific/Wake',
    'America/New_York',
    'America/Detroit',
    'America/Kentucky/Louisville',
    'America/Kentucky/Monticello',
    'America/Indiana/Indianapolis',
    'America/Indiana/Vincennes',
    'America/Indiana/Winamac',
    'America/Indiana/Marengo',
    'America/Indiana/Petersburg',
    'America/Indiana/Vevay',
    'America/Chicago',
    'America/Indiana/Tell_City',
    'America/Indiana/Knox',
    'America/Menominee',
    'America/North_Dakota/Center',
    'America/North_Dakota/New_Salem',
    'America/North_Dakota/Beulah',
    'America/Denver',
    'America/Boise',
    'America/Phoenix',
    'America/Los_Angeles',
    'America/Anchorage',
    'America/Juneau',
    'America/Sitka',
    'America/Metlakatla',
    'America/Yakutat',
    'America/Nome',
    'America/Adak',
    'Pacific/Honolulu',
    'America/Montevideo',
    'Asia/Samarkand',
    'Asia/Tashkent',
    'America/Caracas',
    'Asia/Ho_Chi_Minh',
    'Pacific/Efate',
    'Pacific/Wallis',
    'Pacific/Apia',
    'Africa/Johannesburg'
];

8
这应该在顶部
尤金(Eugene)

请注意,这在某些浏览器中不起作用。例如。IE11。
Paul LeBeau

IE控制台中的错误: “ timeZone”的选项值“ AMERICA / NEW_YORK”超出有效范围。预期:['UTC'] @OloghoCyrilPaul
Matee Gojra

1
很简单,很优雅。您可以在此处找到所有时区的列表stackoverflow.com/questions/38399465/…。对于UTC,请选择伦敦时区。
EPurpl3

1
这些值都不是“时区”,它们是IANA时区数据库的代表位置,用于具有相同的历史本地时区和夏时制更改的地方。
RobG

28

我不认为这是可能的-创建日期对象后,无法在其上设置时区。

从某种意义上讲,这是有道理的-从概念上讲(如果可能在实施中不可行);每个http://en.wikipedia.org/wiki/Unix_timestamp(重点是我的):

Unix时间(或POSIX时间)是一种用于描述时间的系统,时间定义为自1970年1月1日午夜协调世界时(UTC)起经过的秒数。

一旦构建了一个,它将代表“实时”中的某个点。时区仅在您要将抽象时间点转换为人类可读的字符串时才相关。

因此,有意义的是,您只能更改Date在构造函数中表示的实际时间。令人遗憾的是,似乎没有办法传递明确的时区-您正在调用的构造函数(可以说是正确的)在规范存储它们时将“本地”时间变量转换为GMT,因此无法使用int, int, int构造函数用于格林尼治标准时间。

从好的方面来说,仅使用采用String的构造函数是微不足道的。您甚至不必将数字月份转换为字符串(至少在Firefox上),因此我希望可以采用幼稚的实现。但是,在试用之后,它可以在Firefox,Chrome和Opera中成功运行,但在Konqueror(“无效日期”),Safari(“无效日期”)和IE(“ NaN”)中无法运行。我想您只需要一个查找数组即可将月份转换为字符串,如下所示:

var months = [ '', 'January', 'February', ..., 'December'];

function createGMTDate(xiYear, xiMonth, xiDate) {
   return new Date(months[xiMonth] + ' ' + xiDate + ', ' + xiYear + ' 00:00:00 GMT');
}

6
如果没有办法“Date对象上设置的时区创建后”,是不是在暗示,有一个方法来设置Date对象的时区创建它?看起来js日期不是“自大纪元以来大约几秒钟的薄包装纸”-看起来是秒数加上一个时区。
安东尼

1
@Anthony,它只能使用客户端的时区。Javascript可以在utc本地执行,也可以返回本地,但是无法访问时区数据库。例如,它无法告诉您在圣地亚哥时在墨西哥城的时间。
塞缪尔·丹尼尔森


16

如果要处理从年,月,日,……(包括时区)中创建Javascript Date对象的稍微不同但相关的问题,也就是说,如果要将字符串解析为Date,那么您可以显然必须做一个令人难以置信的复杂舞蹈:

// parseISO8601String : string -> Date
// Parse an ISO-8601 date, including possible timezone,
// into a Javascript Date object.
//
// Test strings: parseISO8601String(x).toISOString()
// "2013-01-31T12:34"              -> "2013-01-31T12:34:00.000Z"
// "2013-01-31T12:34:56"           -> "2013-01-31T12:34:56.000Z"
// "2013-01-31T12:34:56.78"        -> "2013-01-31T12:34:56.780Z"
// "2013-01-31T12:34:56.78+0100"   -> "2013-01-31T11:34:56.780Z"
// "2013-01-31T12:34:56.78+0530"   -> "2013-01-31T07:04:56.780Z"
// "2013-01-31T12:34:56.78-0330"   -> "2013-01-31T16:04:56.780Z"
// "2013-01-31T12:34:56-0330"      -> "2013-01-31T16:04:56.000Z"
// "2013-01-31T12:34:56Z"          -> "2013-01-31T12:34:56.000Z"
function parseISO8601String(dateString) {
    var timebits = /^([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2})(?::([0-9]*)(\.[0-9]*)?)?(?:([+-])([0-9]{2})([0-9]{2}))?/;
    var m = timebits.exec(dateString);
    var resultDate;
    if (m) {
        var utcdate = Date.UTC(parseInt(m[1]),
                               parseInt(m[2])-1, // months are zero-offset (!)
                               parseInt(m[3]),
                               parseInt(m[4]), parseInt(m[5]), // hh:mm
                               (m[6] && parseInt(m[6]) || 0),  // optional seconds
                               (m[7] && parseFloat(m[7])*1000) || 0); // optional fraction
        // utcdate is milliseconds since the epoch
        if (m[9] && m[10]) {
            var offsetMinutes = parseInt(m[9]) * 60 + parseInt(m[10]);
            utcdate += (m[8] === '+' ? -1 : +1) * offsetMinutes * 60000;
        }
        resultDate = new Date(utcdate);
    } else {
        resultDate = null;
    }
    return resultDate;
}

也就是说,您使用不带时区的日期创建一个“ UTC时间”(因此您知道它所在的语言环境,即UTC“语言环境”,并且未默认为本地语言环境),然后手动应用指示的时区偏移量。

如果有人实际上考虑了Javascript日期对象超过五分钟,那不是很好吗?


1
感谢您的强大功能!我唯一要更改的是在时区偏移中添加对冒号的支持。var timebits = / ^([0-9] {4})-([0-9] {2})-([0-9] {2})T([0-9] {2}):( [0-9] {2})(?::([0-9] *)(\。[0-9] *)?)?(?:([+-])([0-9] { 2} [:]?)([0-9] {2}))?/;
robnardo

2
他们确实考虑过;不幸的是,“他们”是Java语言设计人员,因为JS只是将Java的Date类复制到其初始实现中。
Xanthir

@Xanthir Oooh,您是对的,我忘记了原始Java Date对象的糟糕程度;但是至少Java弃用了它并继续前进,这似乎是Java脚本无法完成的事情(这是一种奇怪的语言,Java脚本:非常可爱,并且不像它最初看起来的那么糟糕)。
诺曼·格雷

13
d = new Date();
utc = d.getTime() + (d.getTimezoneOffset() * 60000);
nd = new Date(utc + (3600000*offset));

offset value base on which location time zone you would like to set 
For India offset value +5.5,
New York offset value -4,
London offset value +1

对于所有位置偏移量UTC时间偏移量的Wiki列表


在夏令时期间,这对纽约有何帮助?
frederj

对于纽约,请使用偏移值-4
Vijay Lathiya,

1
纽约的偏移量根据夏令时有所不同。有时是-4,有时是-5 en.wikipedia.org/wiki/Eastern_Time_Zone
frederj

8

对于UTC + z,getTimeZoneOffset为负。

var d = new Date(xiYear, xiMonth, xiDate);
if(d.getTimezoneOffset() > 0){
    d.setTime( d.getTime() + d.getTimezoneOffset()*60*1000 );
}

1
一个小错误,!= 0,而不是>0。我最终使用了这个
Cristi Băluță

8

这可能会帮助某人,将UTC放在您传递给新构造函数的末尾

至少在Chrome中您可以说 var date = new Date("2014-01-01 11:00:00 UTC")


1
在Safari上返回“无效日期”
pmrotule '17

1
+0000UTC 替换为(注意,必须删除00和UTC之间的空格),这将在Firefox和Chrome上均适用。不过,不确定使用Safari。(参考:stackoverflow.com/a/17545854/1273587
cytsunny19年

8

一线解决方案

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

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


4
这将是在客户端上运行的代码,这意味着您所在位置不同的用户的时区将有所不同。此解决方案将要求每个需要它的人都住在同一时区(您自己)。
凯文·比尔,2015年

@Kevin Beal在这种情况下,只需使用getTimezoneOffset
maximus

6
// My clock 2018-07-25, 00:26:00 (GMT+7)
let date = new Date(); // 2018-07-24:17:26:00 (Look like GMT+0)
const myTimeZone = 7; // my timeZone 
// my timeZone = 7h = 7 * 60 * 60 * 1000 (millisecond);
// 2018-07-24:17:26:00 = x (milliseconds)
// finally, time in milliseconds (GMT+7) = x + myTimezone 
date.setTime( date.getTime() + myTimeZone * 60 * 60 * 1000 );
// date.toISOString() = 2018-07-25, 00:26:00 (GMT+7)

解释代码及其解决问题的方式将提高答案的质量,并有助于学习用户。
Nic3500 '18

5

我发现获得正确日期的最简单方法是使用datejs。

http://www.datejs.com/

我通过Ajax以这种格式将日期作为字符串获取:'2016-01-12T00:00:00'

var yourDateString = '2016-01-12T00:00:00';
var yourDate = new Date(yourDateString);
console.log(yourDate);
if (yourDate.getTimezoneOffset() > 0){
    yourDate = new Date(yourDateString).addMinutes(yourDate.getTimezoneOffset());
}
console.log(yourDate);

控制台将显示为:

2016年1月11日星期一19:00:00 GMT-0500(东部标准时间)

2016年1月12日,星期二,格林尼治标准时间-0500(东部标准时间)

https://jsfiddle.net/vp1ena7b/3/

“ addMinutes”来自datejs,您可以自己在纯js中完成此操作,但是我的项目中已经有datejs,因此我找到了一种使用它来获取正确日期的方法。

我认为这可能对某人有帮助...


尝试了所有方法,这是我所追求的唯一获得午夜的方法!
SharpC

3

任何里程

var d = new Date(xiYear, xiMonth, xiDate).toLocaleString();

这似乎对我有用(离格林尼治标准时间一个时区),但是由于“语言环境”不一定与时区相关,因此我不会依赖它。
Wytze 2014年

3

此代码将返回以浏览器时区格式设置的Date对象

Date.prototype.timezone = function () {
    this.setHours(this.getHours() + (new Date().getTimezoneOffset() / 60));
    return this;
}

编辑:

为了避免污染Date API,可以将上述函数转换为实用程序函数。该函数接受一个Date对象,并返回一个变异的Date对象。

function setTimeZone(date) {
    date.setHours(date.getHours() + (new Date().getTimezoneOffset() / 60));
    return date;
}

6
不支持扩展本地对象
Paul Rumkin

1

我从中看到的最佳解决方案来自

http://www.codingforums.com/archive/index.php/t-19663.html

打印时间功能

<script language="javascript" type="text/javascript">
//borrowed from echoecho
//http://www.echoecho.com/ubb/viewthread.php?tid=2362&pid=10482&#pid10482
workDate = new Date()
UTCDate = new Date()
UTCDate.setTime(workDate.getTime()+workDate.getTimezoneOffset()*60000)

function printTime(offset) {
    offset++;
    tempDate = new Date()
    tempDate.setTime(UTCDate.getTime()+3600000*(offset))
    timeValue = ((tempDate.getHours()<10) ? ("0"+tempDate.getHours()) : (""+tempDate.getHours()))
    timeValue += ((tempDate.getMinutes()<10) ? ("0"+tempDate.getMinutes()) : tempDate.getMinutes())
    timeValue += " hrs."
    return timeValue
    }
    var now = new Date()
    var seed = now.getTime() % 0xfffffff
    var same = rand(12)
</script>

Banff, Canada:
<script language="JavaScript">document.write(printTime("-7"))</script>

完整代码示例

<html>

<head>
<script language="javascript" type="text/javascript">
//borrowed from echoecho
//http://www.echoecho.com/ubb/viewthread.php?tid=2362&pid=10482&#pid10482
workDate = new Date()
UTCDate = new Date()
UTCDate.setTime(workDate.getTime()+workDate.getTimezoneOffset()*60000)

function printTime(offset) {
offset++;
tempDate = new Date()
tempDate.setTime(UTCDate.getTime()+3600000*(offset))
timeValue = ((tempDate.getHours()<10) ? ("0"+tempDate.getHours()) : (""+tempDate.getHours()))
timeValue += ((tempDate.getMinutes()<10) ? ("0"+tempDate.getMinutes()) : tempDate.getMinutes())
timeValue += " hrs."
return timeValue
}
var now = new Date()
var seed = now.getTime() % 0xfffffff
var same = rand(12)
</script>

</head>

<body>
Banff, Canada:
<script language="JavaScript">document.write(printTime("-7"))</script>
<br>
Michigan:
<script language="JavaScript">document.write(printTime("-5"))</script>
<br>
Greenwich, England(UTC):
<script language="JavaScript">document.write(printTime("-0"))</script>
<br>
Tokyo, Japan:
<script language="JavaScript">document.write(printTime("+9"))</script>
<br>
Berlin, Germany:
<script language="JavaScript">document.write(printTime("+1"))</script>

</body>
</html>

您的示例不包括夏令时。当前时间:2013年10月4日星期五格林尼治标准时间0700(太平洋夏令时)UTC时间:2013年10月4日星期五,加拿大格林尼治标准时间班夫,加拿大:1213小时。密西根州:1413小时。英国格林威治(UTC):1913小时。日本东京:0413小时。德国柏林:2013小时。
Jeson Martajaya 2013年

0

如果要检查两个日期之间的时差,只需检查第二个时区是比第一个所需时区小还是大,然后减去或增加一个时间即可。

  const currTimezone = new Date().getTimezoneOffset(); // your timezone
  const newDateTimezone = date.getTimezoneOffset(); // date with unknown timezone

  if (currTimezone !== newDateTimezone) {
    // and below you are checking if difference should be - or +. It depends on if unknown timezone is lesser or greater than yours
    const newTimezone = (currTimezone - newDateTimezone) * (currTimezone > newDateTimezone ? 1 : -1);
    date.setTime(date.getTime() + (newTimezone * 60 * 1000));
  }

0

GMT -03:00示例

new Date(new Date()-3600*1000*3).toISOString();  // 2020-02-27T15:03:26.261Z

甚至

now  = new Date().getTime()-3600*1000*3; // 1582818380528
data = new Date(now).toISOString();      // 2020-02-27T15:03:26.261Z

-1

这对我有用。不确定这是否是个好主意。

var myDate = new Date();
console.log('myDate:', myDate);   // myDate: "2018-04-04T01:09:38.112Z"

var offset = '+5';  // e.g. if the timeZone is -5

var MyDateWithOffset = new Date( myDate.toGMTString() + offset );   

console.log('MyDateWithOffset:', MyDateWithOffset); // myDateWithOffset: "2018-04-03T20:09:38.000Z"


-1

我使用了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());
},

-11

这是最好的解决方案

使用:

// TO ALL dates
Date.timezoneOffset(-240) // +4 UTC

// Override offset only for THIS date
new Date().timezoneOffset(-180) // +3 UTC

码:

Date.prototype.timezoneOffset = new Date().getTimezoneOffset();

Date.setTimezoneOffset = function(timezoneOffset) {
  return this.prototype.timezoneOffset = timezoneOffset;
};

Date.getTimezoneOffset = function() {
  return this.prototype.timezoneOffset;
};

Date.prototype.setTimezoneOffset = function(timezoneOffset) {
  return this.timezoneOffset = timezoneOffset;
};

Date.prototype.getTimezoneOffset = function() {
  return this.timezoneOffset;
};

Date.prototype.toString = function() {
  var offsetDate, offsetTime;
  offsetTime = this.timezoneOffset * 60 * 1000;
  offsetDate = new Date(this.getTime() - offsetTime);
  return offsetDate.toUTCString();
};

['Milliseconds', 'Seconds', 'Minutes', 'Hours', 'Date', 'Month', 'FullYear', 'Year', 'Day'].forEach((function(_this) {
  return function(key) {
    Date.prototype["get" + key] = function() {
      var offsetDate, offsetTime;
      offsetTime = this.timezoneOffset * 60 * 1000;
      offsetDate = new Date(this.getTime() - offsetTime);
      return offsetDate["getUTC" + key]();
    };
    return Date.prototype["set" + key] = function(value) {
      var offsetDate, offsetTime, time;
      offsetTime = this.timezoneOffset * 60 * 1000;
      offsetDate = new Date(this.getTime() - offsetTime);
      offsetDate["setUTC" + key](value);
      time = offsetDate.getTime() + offsetTime;
      this.setTime(time);
      return time;
    };
  };
})(this));

咖啡版本:

Date.prototype.timezoneOffset = new Date().getTimezoneOffset()


Date.setTimezoneOffset = (timezoneOffset)->
    return @prototype.timezoneOffset = timezoneOffset


Date.getTimezoneOffset = ->
    return @prototype.timezoneOffset


Date.prototype.setTimezoneOffset = (timezoneOffset)->
    return @timezoneOffset = timezoneOffset


Date.prototype.getTimezoneOffset = ->
    return @timezoneOffset


Date.prototype.toString = ->
    offsetTime = @timezoneOffset * 60 * 1000
    offsetDate = new Date(@getTime() - offsetTime)
    return offsetDate.toUTCString()


[
    'Milliseconds', 'Seconds', 'Minutes', 'Hours',
    'Date', 'Month', 'FullYear', 'Year', 'Day'
]
.forEach (key)=>
    Date.prototype["get#{key}"] = ->
        offsetTime = @timezoneOffset * 60 * 1000
        offsetDate = new Date(@getTime() - offsetTime)
        return offsetDate["getUTC#{key}"]()

    Date.prototype["set#{key}"] = (value)->
        offsetTime = @timezoneOffset * 60 * 1000
        offsetDate = new Date(@getTime() - offsetTime)
        offsetDate["setUTC#{key}"](value)
        time = offsetDate.getTime() + offsetTime
        @setTime(time)
        return time

2
哇,我也不喜欢,但是我想人们真的很讨厌您覆盖内置的原型!
来自Qaribou的Josh,18年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.