如何使用Java处理日历时区?

2022-08-31 10:44:21

我有一个来自我的应用程序的时间戳值。用户可以位于任何给定的本地时区。

由于此日期用于假设给定时间始终以 GMT 为单位的 Web 服务,因此我需要将用户的参数从 say (EST) 转换为 (GMT)。这是踢球者:用户忘记了他的TZ。他输入了要发送到 WS 的创建日期,因此我需要的是:

用户输入:5/1/2008 6:12 PM (EST)
WS 的参数需要为:5/1/2008 6:12 PM (GMT)

我知道默认情况下,时间戳总是应该在GMT中,但是当发送参数时,即使我从TS(应该是在GMT中)创建了我的日历,除非用户处于GMT,否则小时数始终关闭。我错过了什么?

Timestamp issuedDate = (Timestamp) getACPValue(inputs_, "issuedDate");
Calendar issueDate = convertTimestampToJavaCalendar(issuedDate);
...
private static java.util.Calendar convertTimestampToJavaCalendar(Timestamp ts_) {
  java.util.Calendar cal = java.util.Calendar.getInstance(
      GMT_TIMEZONE, EN_US_LOCALE);
  cal.setTimeInMillis(ts_.getTime());
  return cal;
}

使用前面的代码,这就是我得到的结果(为了便于阅读,短格式):

[2008年5月1日 下午11:12]


答案 1
public static Calendar convertToGmt(Calendar cal) {

    Date date = cal.getTime();
    TimeZone tz = cal.getTimeZone();

    log.debug("input calendar has date [" + date + "]");

    //Returns the number of milliseconds since January 1, 1970, 00:00:00 GMT 
    long msFromEpochGmt = date.getTime();

    //gives you the current offset in ms from GMT at the current date
    int offsetFromUTC = tz.getOffset(msFromEpochGmt);
    log.debug("offset is " + offsetFromUTC);

    //create a new calendar in GMT timezone, set to this date and add the offset
    Calendar gmtCal = Calendar.getInstance(TimeZone.getTimeZone("GMT"));
    gmtCal.setTime(date);
    gmtCal.add(Calendar.MILLISECOND, offsetFromUTC);

    log.debug("Created GMT cal with date [" + gmtCal.getTime() + "]");

    return gmtCal;
}

如果我在 中传递当前时间(“12:09:05 EDT”来自 ),则输出如下:Calendar.getInstance()

调试 - 输入日历有日期 [星期四 十月 23 12:09:05 EDT 2008]
调试 - 偏移量为 -14400000
DEBUG - 创建 GMT cal 与日期 [星期四 十月 23 08:09:05 EDT 2008]

12:09:05 GMT为8:09:05 EDT。

这里令人困惑的部分是,它会返回当前时区的a,并且没有办法修改日历的时区并滚动基础日期。根据 Web 服务采用的参数类型,您可能只想以 epoch 的毫秒数处理 WS。Calendar.getTime()Date


答案 2

谢谢大家的回复。经过进一步调查,我得到了正确的答案。正如Skip Head所提到的,我从应用程序获得的时间戳正在调整为用户的时区。因此,如果用户输入下午6:12(美国东部标准时间),我将得到下午2:12(GMT)。我需要的是一种撤消转换的方法,以便用户输入的时间是我发送到WebServer请求的时间。以下是我完成此操作的方式:

// Get TimeZone of user
TimeZone currentTimeZone = sc_.getTimeZone();
Calendar currentDt = new GregorianCalendar(currentTimeZone, EN_US_LOCALE);
// Get the Offset from GMT taking DST into account
int gmtOffset = currentTimeZone.getOffset(
    currentDt.get(Calendar.ERA), 
    currentDt.get(Calendar.YEAR), 
    currentDt.get(Calendar.MONTH), 
    currentDt.get(Calendar.DAY_OF_MONTH), 
    currentDt.get(Calendar.DAY_OF_WEEK), 
    currentDt.get(Calendar.MILLISECOND));
// convert to hours
gmtOffset = gmtOffset / (60*60*1000);
System.out.println("Current User's TimeZone: " + currentTimeZone.getID());
System.out.println("Current Offset from GMT (in hrs):" + gmtOffset);
// Get TS from User Input
Timestamp issuedDate = (Timestamp) getACPValue(inputs_, "issuedDate");
System.out.println("TS from ACP: " + issuedDate);
// Set TS into Calendar
Calendar issueDate = convertTimestampToJavaCalendar(issuedDate);
// Adjust for GMT (note the offset negation)
issueDate.add(Calendar.HOUR_OF_DAY, -gmtOffset);
System.out.println("Calendar Date converted from TS using GMT and US_EN Locale: "
    + DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.SHORT)
    .format(issueDate.getTime()));

代码的输出为:(用户输入 5/1/2008 6:12PM (EST)

当前用户的时区:EST
当前与 GMT 的偏移量(以小时为单位):-4(通常为 -5,DST 调整后除外)
来自 ACP 的 TS:2008-05-01 14:12:00.0
使用 GMT 和 US_EN从 TS 转换的日历日期 区域设置:5/1/08 下午 6:12 (GMT)


推荐