Pat*_*rts 1 javascript date dst timezone-offset
我已经回顾了几个关于类似主题的问题,但没有一个问题涉及计算目的地时区,将其夏令时(夏令时)考虑在内.我正在尝试编写一个简单的小部件,向访问该页面的任何人显示特定时区的实时本地时间.感兴趣的时区是BST(什么是BST?),但如果可能的话,我很乐意看到任何语言环境的通用实现.这是我使用vanilla JavaScript的尝试:
function getBST() {
var date = new Date(),
utc = date.getTime() + date.getTimezoneOffset() * 60000,
local = new Date(utc), // this is GMT, not BST
day = local.getDate(),
mon = local.getMonth() + 1,
year = local.getFullYear(),
hour = local.getHours(),
minute = ('0' + local.getMinutes()).slice(-2),
second = ('0' + local.getSeconds()).slice(-2),
suffix = hour < 12 ? 'AM' : 'PM';
hour = (hour - 24) % 12 + 12;
return mon + '/' + day + '/' + year + ', ' + hour + ':' + minute + ':' + second + ' ' + suffix;
}
setInterval(function () {
document.body.textContent = getBST();
}, 1000);
Run Code Online (Sandbox Code Playgroud)
根据@ RobG的回答,我写了这段代码:
function resetDay(date) {
date.setUTCMilliseconds(0);
date.setUTCSeconds(0);
date.setUTCMinutes(0);
date.setUTCHours(1);
return date;
}
function lastDay(date, day) {
while (date.getUTCDay() !== day) {
date.setUTCDate(date.getUTCDate() - 1);
}
return date;
}
function adjustDST(date, begin, end) {
if (date >= begin && date < end) {
date.setUTCHours(date.getUTCHours() + 1);
}
return date;
}
function updateBST() {
var date = new Date();
var begin = resetDay(new Date());
begin.setUTCMonth(3, -1);
begin = lastDay(begin, 0);
var end = resetDay(new Date());
end.setUTCMonth(10, -1);
end = lastDay(end, 0);
date = adjustDST(date, begin, end);
var day = date.getUTCDate(),
mon = date.getUTCMonth() + 1,
year = date.getUTCFullYear(),
hour = date.getUTCHours(),
minute = ('0' + date.getUTCMinutes()).slice(-2),
second = ('0' + date.getUTCSeconds()).slice(-2),
suffix = hour < 12 ? 'AM' : 'PM';
hour = (hour - 24) % 12 + 12;
return mon + '/' + day + '/' + year + ', ' + hour + ':' + minute + ':' + second + ' ' + suffix;
}
setInterval(function () {
document.body.textContent = updateBST();
}, 1000);
Run Code Online (Sandbox Code Playgroud)
我通过在调试器中暂停并更改当前日期的月份来测试BST,并且输出是一小时后因此它似乎正常工作.感谢大家的帮助!
在纯JS中,仅使用这些Date
方法是不可能的,但是(当然)有一个lib:http://momentjs.com/timezone/
例:
moment.tz("Europe/London").format(); // 2016-01-15T09:21:08-07:00
Run Code Online (Sandbox Code Playgroud)
支持一个时区并不难,只要您知道它何时进入和退出夏令时。
一个 javascript 日期由一个 UTC 时间值和一个基于主机系统设置的时区偏移量组成。所以你需要做的就是将你想要的时区偏移量应用到 UTC 时间,然后很快,任何时区都有你的时间。
虽然有一些事实上的标准(例如IATA 时区代码和IANA 时区),但没有用于缩写时区的标准化系统。我想 BST 是指英国夏令时,也称为英国夏令时 (BDT) 和英国夏令时 (BDST)。它也可能是孟加拉国标准时间或布干维尔标准时间,也称为“BST”。
有各种库(例如Moment Timezone)使用 IANA 代码,并且可以在任何时间(或多或少)提供任何受支持时区的时间。
BST 于每年 3 月最后一个星期日的 01:00 UTC 开始,并于每年 10 月最后一个星期日的 01:00 UTC 结束,因此算法为:
new Date()
)就是这样,唯一困难的部分是找到合适的星期日日期。您根本不需要考虑本地时区偏移量,因为一切都基于 UTC,而 Date 对象也是如此。
现在我没有时间提供代码,所以请试一试,我会在大约 10 小时内回复您。
所以这是代码。前两个函数是辅助函数,一个获取一个月中的最后一个星期日,另一个使用偏移格式化 ISO 8601 字符串。大部分工作都在这两个函数中。希望评论足够,如果需要更多解释,请询问。
我没有在字符串中包含毫秒,如果需要,可以随意添加它们+ '.' + ('00' + d.getUTCMilliseconds()).slice(-3)
,在格式化字符串的偏移部分之前添加。
请注意,如果开始或停止夏令时的日期发生更改,则需要修改该函数,但这种情况很少见。历史日期当然需要一个小型数据库,其中包含特定年份和时期夏令时开始和停止的时间。
/* Return a Date for the last Sunday in a month
** @param {number} year - full year number (e.g. 2015)
** @param {number} month - calendar month number (jan=1)
** @returns {Date} date for last Sunday in given month
*/
function getLastSunday(year, month) {
// Create date for last day in month
var d = new Date(year, month, 0);
// Adjust to previous Sunday
d.setDate(d.getDate() - d.getDay());
return d;
}
/* Format a date string as ISO 8601 with supplied offset
** @param {Date} date - date to format
** @param {number} offset - offset in minutes (+east, -west), will be
** converted to +/-00:00
** @returns {string} formatted date and time
**
** Note that javascript Date offsets are opposite: -east, +west but
** this function doesn't use the Date's offset.
*/
function formatDate(d, offset) {
function z(n){return ('0'+n).slice(-2)}
// Default offset to 0
offset = offset || 0;
// Generate offset string
var offSign = offset < 0? '-' : '+';
offset = Math.abs(offset);
var offString = offSign + ('0'+(offset/60|0)).slice(-2) + ':' + ('0'+(offset%60)).slice(-2);
// Generate date string
return d.getUTCFullYear() + '-' + z(d.getUTCMonth()+1) + '-' + z(d.getUTCDate()) +
'T' + z(d.getUTCHours()) + ':' + z(d.getUTCMinutes()) + ':' + z(d.getUTCSeconds()) +
offString;
}
/* Return Date object for current time in London. Assumes
** daylight saving starts at 01:00 UTC on last Sunday in March
** and ends at 01:00 UTC on the last Sunday in October.
** @param {Date} d - date to test. Default to current
** system date and time
** @param {boolean, optional} obj - if true, return a Date object. Otherwise, return
** an ISO 8601 formatted string
*/
function getLondonTime(d, obj) {
// Use provided date or default to current date and time
d = d || new Date();
// Get start and end dates for daylight saving for supplied date's year
// Set UTC date values and time to 01:00
var dstS = getLastSunday(d.getFullYear(), 3);
var dstE = getLastSunday(d.getFullYear(), 10);
dstS = new Date(Date.UTC(dstS.getFullYear(), dstS.getMonth(), dstS.getDate(),1));
dstE = new Date(Date.UTC(dstE.getFullYear(), dstE.getMonth(), dstE.getDate(),1));
// If date is between dstStart and dstEnd, add 1 hour to UTC time
// and format using +60 offset
if (d > dstS && d < dstE) {
d.setUTCHours(d.getUTCHours() +1);
return formatDate(d, 60);
}
// Otherwise, don't adjust and format with 00 offset
return obj? d : formatDate(d);
}
document.write('Current London time: ' + getLondonTime(new Date()));
Run Code Online (Sandbox Code Playgroud)