Создать дату с заданным часовым поясом без использования строкового представления - PullRequest
373 голосов
/ 13 января 2009

У меня есть веб-страница с тремя выпадающими списками для дня, месяца и года. Если я использую конструктор JavaScript Date, который принимает числа, то я получаю объект Date для моего текущего часового пояса:

new Date(xiYear, xiMonth, xiDate)

Дайте правильную дату, но она думает, что это время по Гринвичу + 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

что я и хотел.

Ответы [ 19 ]

447 голосов
/ 13 января 2009

с использованием .setUTCHours() было бы возможно установить даты в UTC-времени, что позволило бы вам использовать UTC-время во всей системе.

Вы не можете установить его, используя UTC в конструкторе, если вы не укажете строку даты.

Используя new Date(Date.UTC(year, month, day, hour, minute, second)), вы можете создать объект Date из определенного времени UTC.

187 голосов
/ 17 апреля 2013
var d = new Date(xiYear, xiMonth, xiDate);
d.setTime( d.getTime() + d.getTimezoneOffset()*60*1000 );

Этот ответ специально составлен для исходного вопроса и не даст ответа, которого вы обязательно ожидаете. В частности, некоторые люди захотят вычесть смещение часового пояса, а не добавить его. Помните, однако, что весь смысл этого решения состоит в том, чтобы взломать объект даты javascript для конкретной десериализации, а не во всех случаях.

157 голосов
/ 23 декабря 2012

Полагаю, вам нужна функция 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()); 
}
25 голосов
/ 13 января 2009

Я не верю, что это возможно - нет возможности установить часовой пояс для объекта Date после его создания.

И в некотором смысле это имеет смысл - концептуально (если, возможно, не в реализации); за http://en.wikipedia.org/wiki/Unix_timestamp (выделено мое):

Время Unix, или время POSIX, представляет собой систему для описания моментов во времени, определяемых как количество секунд, прошедших с полуночи Всемирное координированное время (UTC) от четверга, 1 января 1970 года.

Как только вы его построите, он будет представлять определенную точку в «реальном» времени. Часовой пояс имеет значение только тогда, когда вы хотите преобразовать этот абстрактный момент времени в удобочитаемую строку.

Таким образом, имеет смысл, что вы сможете изменить только фактическое время, которое Date представляет в конструкторе. К сожалению, кажется, что нет никакого способа передать в явном часовом поясе - и конструктор, который вы вызываете (возможно, правильно), переводит ваши «локальные» переменные времени в GMT, когда он канонически сохраняет их - поэтому нет способа использовать int, int, int конструктор для времени по Гринвичу.

С другой стороны, тривиально просто использовать конструктор, который принимает строку. Вам даже не нужно конвертировать числовой месяц в строку (по крайней мере, в 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');
}
17 голосов
/ 19 сентября 2014

Я знаю, что это старо, но если это поможет, вы можете использовать момент и часовой пояс. Если вы не видели их, посмотрите.

http://momentjs.com/timezone/

http://momentjs.com/

две действительно удобные библиотеки управления временем.

16 голосов
/ 01 февраля 2013

Если вы хотите решить немного другую, но связанную проблему создания объекта Javascript 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 более, чем, о-о-о, пять минут ....

15 голосов
/ 31 января 2019

Просто установите часовой пояс и вернитесь согласно

new Date().toLocaleString("en-US", {timeZone: "America/New_York"})
13 голосов
/ 10 июня 2016
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

для всех смещений местоположения Wiki-список смещений времени UTC

9 голосов
/ 29 января 2015

Однолинейное решение

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

Вместо 1422524805305 используйте метку времени в миллисекундах Вместо 330 используйте смещение часового пояса в минутах. Время по Гринвичу (например, Индия +5: 30 - 5 * 60 + 30 = 330 минут)

8 голосов
/ 28 августа 2014

Это может помочь кому-то, поставить UTC в конце того, что вы передаете новому конструктору

По крайней мере, в хром можно сказать var date = new Date("2014-01-01 11:00:00 UTC")

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...