【Android 国际化】碰到有关日期的时区问题,我们该怎么办?终极大招来了。。

背景

为什么想到要写一篇关于时区问题的blog?因为。。
去年Colin参与开发了公司的一个独立app的开发,其中我只负责了“多目的地选择页”的开发,说白了就是去打了个酱油。。可这酱油还是没完全白打,我发现项目经理、产品、开发、测试们在项目后期都在纠结时区问题。虽然我也跟着开了几场关于时区问题的会,可由于我没有真正参与吧,并没有真正理解为啥时区会出问题。。
不过Colin从那时就告诉自己,下次等我遇到时区问题时,一定要好好研究,这块内容貌似挺有意思的。。
就这样,真让我等来了。。

为什么会有时区问题?

  1. 国际化app中需要展示时间时
  2. app中用到了Calendar类orDate类时

总的来说,当需要把Date转化成时间字符串时,还有就是用到了Calendar类来计算时间时,如果用户在国外使用你的app时就可能存在时区问题。这里说的时区问题到底是什么问题呢?
比如,北京时间5月23号5:30时,另外一个半球的国家很有可能是5月22号晚上,那么一个字符串“2018-5-23 5:30”经过转为Date就转成了当地的Date,当这个Date再转成字符串时就不再是“2018-5-23 5:30”了。

该如何解决时区问题?

1. 关于时区的几个基本概念

要想彻底理解时区问题,我们需要先了解几个基本概念。

1.协调世界时:Coordinated Universal Time(UTC):
是英国格林尼治时间,格林尼治时间被称为零时区时间,也被称为绝对时间。
2.格林尼治标准时:Greenwich Mean Time(GMT):
是指位于伦敦郊区的皇家格林尼治天文台的标准时间,由于地球在它的椭圆轨道里的运动速度不均匀,这个时刻可能和实际的太阳时相差16分钟。 现在都采用UTC时间,是校正过的GMT时间。

2. Android中处理时区问题的办法

把当地时间都转为北京时间处理,这里有两个办法

(1)全局设置app时区为北京时区
TimeZone chinaTimeZone = TimeZone.getTimeZone("GMT+8");
TimeZone.setDefault(chinaTimeZone);

这个方法比较暴力,我们之前的独立app就是采用这种办法。不过,考虑到可能影响其他页面的功能,我没有采用这种办法。

(2)在需要处理时间时,把本地时区的Date转为北京地区的Date

//转换任意时区的办法(本篇blog精髓)
private fun changeTimeZone(date: Date, oldZone: TimeZone, newZone: TimeZone): Date {
        var dateTmp: Date
        val timeOffset = oldZone.rawOffset - newZone.rawOffset
        dateTmp = Date(date.time - timeOffset)
        return dateTmp
}
//将当地时间转为北京时间的方法
fun getDateOfBJTimeZone(date: Date): Date {
        var originTimeZone = TimeZone.getDefault()
        var bjTimeZone = TimeZone.getTimeZone("Asia/Shanghai")
        return changeTimeZone(date, originTimeZone, bjTimeZone)
}

上面提供了两个Date转换的方法,那需要用到Calendar类时,可以给Calendar设置一个时区的,办法如下:

val c = Calendar.getInstance()
c.timeZone = TimeZone.getTimeZone("Asia/Shanghai")
c.timeInMillis = date.time
c.add(Calendar.DATE, days)

同样,String和Date互转时也可以设置时区,办法如下:

val dateFormat = SimpleDateFormat(type)
val timeZone = TimeZone.getTimeZone("Asia/Shanghai")
dateFormat.timeZone = timeZone
str = dateFormat.format(date)

参考

http://www.cnblogs.com/endure/p/3485281.html
https://www.jianshu.com/p/098c16b39f9b
https://blog.csdn.net/l_serein/article/details/6273362
http://www.cnblogs.com/endure/p/3485281.html
https://www.cnblogs.com/wenfei123chai/p/4267998.html
https://blog.csdn.net/lovext4098477/article/details/46866879

猜你喜欢

转载自blog.csdn.net/colinandroid/article/details/80428350