【问题标题】:Showing correct time from Milliseconds with desired TimeZone从所需的 TimeZone 显示毫秒的正确时间
【发布时间】:2018-02-16 00:08:48
【问题描述】:

我正在开发一个从Google TimeZone API 获取数据的应用程序。 只是我有时间在地球上所需的地方,以毫秒为单位。

例如:1504760156000 它显示的是伦敦的日期时间

2017 年 9 月 7 日星期四 09:55:56

由于我在 UTC 的 TimeZone +05:00 中,如果我操纵 1504760156000 这些毫秒,它将显示整个日期时间,如下所示:

2017 年 9 月 7 日星期四 09:55:56 GMT+0500(巴基斯坦标准时间)

但我想展示:

2017 年 9 月 7 日星期四 09:55:56 GMT+0100(英国夏令时间)

问题是:我有正确的伦敦日期和时间,但可以在不更改时间的情况下显示/更改时区,因为伦敦的时间是正确的。

更新

在得到一些 cmets 之后。在我的例子中你没有让我在这里。

假设我在巴基斯坦,这里的时间是1:55 PM,所以我通过我的应用程序询问 GOOGLE API,告诉我现在伦敦的时间是几点。 Google API 告诉我在伦敦的时间是 1504760156000(上午 9 点 55 分),如果我将这些毫秒转换为 Date 对象,它将打印如下:

Date date =new Date(1504760156000)

2017 年 9 月 7 日星期四 09:55:56 GMT+0500(巴基斯坦标准时间)

它将根据我的本地时区进行操作,但我想要如下结果

2017 年 9 月 7 日星期四 09:55:56 GMT+0100(英国夏令时间)

更新 2

我在 UTC 中准备了以秒为单位的时间戳,因为 Google 时区 API 需要以秒为单位的时间戳 UTC

"https://maps.googleapis.com/maps/api/timezone/json?location="+Latitude +","+Longitude+"&timestamp="+currentTimeUTCinSeonds+"&key="API KEY"

Google API 以针对伦敦的以下 JSON 响应我。

{
    "dstOffset" : 3600,   
    "rawOffset" : 0,  
    "status" : "OK",  
    "timeZoneId" : "Europe/London",  
    "timeZoneName" : "British Summer Time"
}

根据文档:

计算当地时间

给定位置的当地时间是时间戳的总和 参数,以及结果中的 dstOffset 和 rawOffset 字段。

我总结结果timestamp+rawoffset+dstoffset*1000='1504760156000'(在我尝试的那一刻)

项目代码

Long ultimateTime=((Long.parseLong(timeObject1.getDstOffset())*1000)+(Long.parseLong(timeObject1.getRawOffset())*1000)+timestamp*1000);
                    timeObject1.setTimestamp(ultimateTime);  //its Sime POJO object to save current time of queried Location 
                    Date date=new Date(ultimateTime);
                    date1.setText("Date Time : "+date);

正如我所说,我正在处理本地时区的结果,所以当时它给了我以下结果:

2017 年 9 月 7 日星期四 09:55:56 GMT+0500(巴基斯坦标准时间)

但我知道 API 给了我正确的时间。问题是与 UTC 的本地偏移量。我只想把GMT+0500改成GMT+0100

【问题讨论】:

  • this 的回答可能会对你有所帮助。
  • @kcg 不,你没有得到我的问题。我想在所需的时区中获得结果,而不是在我自己的时区中
  • 但时间戳 1504760156000 相当于巴基斯坦的 09:55 和伦敦的 05:55。在伦敦显示为 09:55 是不正确的。
  • @Hugo,no, timestamp 1504760156000 是我从 Google API 获取的伦敦时间,但我在巴基斯坦,如果我在本地系统上操作它,它将获得默认 +5 TimeZone 但想显示伦敦时间区域在巴基斯坦。仔细阅读我的问题
  • @Zaid Mizra ,那你最好把巴基斯坦当地时间转换成伦敦时间.....

标签: android datetime google-api timezone timezone-offset


【解决方案1】:

时间戳表示自纪元以来经过的时间的“绝对”值。例如,您的 currentTimeUTCinSeconds 表示自 unix 纪元以来的秒数(即 1970-01-01T00:00Z,或 1970 年 1 月 1 日st UTC 午夜)。 Java API 通常使用自纪元以来的毫秒数。

但概念是一样的——这些价值观是“绝对的”:它们对世界上的每个人都是一样的,无论他们身在何处。如果世界不同地区(不同时区)的 2 个人同时获得当前时间戳,他们将获得相同的数字。

不同的是,在不同的时区,相同的数字代表不同的本地日期和时间。

例如,您使用的时间戳对应于 2017 年 9 月 7 日th UTC 08:55:56,其值为 1504774556(自纪元以来的秒数)。这个相同的数字对应于伦敦的 09:55、卡拉奇的 13:55、东京的 17:55 等等。 更改此数字将更改每个人的当地时间 - 无需操纵它。

如果你想得到一个代表这个瞬间的java.util.Date,只需这样做:

int currentTimeUTCinSeconds = 1504774556;
// cast to long to not lose precision
Date date = new Date((long) currentTimeUTCinSeconds * 1000);

此日期将保留值 1504774556000(自纪元以来的毫秒数)。该值对应于伦敦的 09:55、卡拉奇的 13:55 和东京的 17:55。

但打印此日期会将其转换为您的 JVM 默认时区(here 是对 Date::toString() 方法行为的一个很好的解释)。当你执行"Date Time : "+date 时,它会隐式调用toString() 方法,结果是转换为默认时区的日期。

如果您希望日期采用特定格式和特定​​时区,则需要SimpleDateFormat。仅打印日期(使用System.out.println 或通过记录它)是行不通的:您无法更改日期对象本身的格式,因为Date has no format

我还使用java.util.Locale 来指定月份和星期几必须是英文。如果您不指定区域设置,它将使用系统默认设置,并且不保证始终为英语(即使在运行时也可以更改,因此最好始终指定区域设置):

// use the same format, use English for month and day of week
SimpleDateFormat sdf = new SimpleDateFormat("EEE MMM dd yyyy HH:mm:ss 'GMT'Z (zzzz)", Locale.ENGLISH);
// set the timezone I want
sdf.setTimeZone(TimeZone.getTimeZone("Europe/London"));
// format the date
System.out.println(sdf.format(date));

输出将是:

2017 年 9 月 7 日星期四 09:55:56 GMT+0100(英国夏令时间)

请注意,我不需要操纵时间戳值。我不使用 google API,但我认为他们的解释太混乱了,上面的代码实现了相同的结果,但复杂性更小。

在您的具体情况下,您可以这样做:

date1.setText("Date Time : "+sdf.format(date));

Java 新的日期/时间 API

旧的类(DateCalendarSimpleDateFormat)有 lots of problemsdesign issues,它们正在被新的 API 取代。

在 Android 中,您可以使用 ThreeTen Backport,这是 Java 8 新日期/时间类的一个很好的反向移植。要使其工作,您还需要ThreeTenABP(更多关于如何使用它here)。

要从时间戳中获取日期,我使用 org.threeten.bp.Instantorg.threeten.bp.ZoneId 将其转换为时区,创建 org.threeten.bp.ZonedDateTime。然后我使用org.threeten.bp.format.DateTimeFormatter 对其进行格式化:

int currentTimeUTCinSeconds = 1504774556;
// get the date in London from the timestamp
ZonedDateTime z = Instant.ofEpochSecond(currentTimeUTCinSeconds).atZone(ZoneId.of("Europe/London"));
// format it
DateTimeFormatter fmt = DateTimeFormatter.ofPattern("EEE MMM dd yyyy HH:mm:ss 'GMT'XX (zzzz)", Locale.ENGLISH);
System.out.println(fmt.format(z));

输出是一样的:

2017 年 9 月 7 日星期四 09:55:56 GMT+0100(英国夏令时间)

在你的情况下,只需这样做:

date1.setText("Date Time : "+fmt.format(z));

【讨论】:

  • 现在你找到我了。这只是打印问题。如果我不指定 Locale.English 怎么办?为什么你指定它?
  • @ZaidMirza 如果不指定语言环境,它将使用 JVM 的默认值(例如,在我使用的 JVM 中,默认值是葡萄牙语),这会影响月份、星期几,甚至时区名称。只需在没有语言环境的情况下进行测试,看看你会得到什么。但是,如果您想要以特定语言输出,最好使用特定语言环境。如果我在 JVM 中使用默认语言环境,则输出为葡萄牙语:Qui set 07 2017 09:55:56 GMT+0100 (Fuso horário de verão da Grã-Bretanha)
  • 谢谢你,先生,你帮了我很多。你让我更清楚我的概念。
  • 这真的很有帮助!谢谢!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-09-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-12-09
  • 1970-01-01
相关资源
最近更新 更多