java中转换不同时区的时间

java中转换不同时区的时间

一丶时区

  由于世界各国家与地区经度不同,地方时也有所不同,因此会划分为不同的时区。

 

  地球是自西向东自转,东边比西边先看到太阳,东边的时间也比西边的早。东边时刻与西边时刻的差值不仅要以时计,而且还要以分和秒来计算,这给人们带来不便。

  为了克服时间上的混乱,1884年在华盛顿召开的一次国际经度会议(又称国际子午线会议)上,规定将全球划分为24个时区(东、西各12个时区)。规定英国(格林尼治天文台旧址)为中时区(零时区)、东1—12区,西1—12区。每个时区横跨经度15度,时间正好是1小时。最后的东、西第12区各跨经度7.5度,以东、西经180度为界。每个时区的中央经线上的时间就是这个时区内统一采用的时间,称为区时,相邻两个时区的时间相差1小时。

 

  计算的区时=已知区时-(已知区时的时区-要计算区时的时区)。(注:东时区为正,西时区为负)

  例1:已知东京(东九区)时间为5月1日12:00,求北京(东八区)的区时。
  北京时间=12:00-(9-8)=11:00,即北京时间为5月1日11:00。
  例2:已知北京时间为5月1日12:00,求伦敦(中时区)的区时。
  伦敦时间=12:00-(8-0)=4:00,即伦敦时间为5月1日4:00。

 

 

二丶UTC时间 与 格林尼治时间

  协调世界时,又称世界统一时间、世界标准时间、国际协调时间。由于英文(CUT)和法文(TUC)的缩写不同,作为妥协,简称UTC。

  格林尼治标准时间(Greenwich Mean Time,GMT)是指位于伦敦郊区的皇家格林尼治天文台的标准时间,因为本初子午线被定义在通过那里的经线。 理论上来说,格林尼治标准时间的正午是指当太阳横穿格林尼治子午线时的时间。由于地球在它的椭圆轨道里的运动速度不均匀,这个时刻可能和实际的太阳时相差16分钟。 地球每天的自转是有些不规则的,而且正在缓慢减速。所以,格林尼治时间已经不再被作为标准时间使用。现在的标准时间——协调世界时(UTC)——由原子钟提供。 自1924年2月5日开始,格林尼治天文台每隔一小时会向全世界发放调时信息。而UTC是基于标准的GMT提供的准确时间。

  GMT(Greenwich Mean Time)——格林尼治标准时间,格林尼治标准时间是19 世纪中叶大英帝国的基准时间,同时也是事实上的世界基准时间。当时主要为了1840 年之后的铁路系统服务。它以格林尼治天文台的经线为0 度经线,将世界分为24 个时区。为了方便,在不需要精确到秒的情况下,通常将GMT 和UTC 视作等同。但UTC 更加科学更加精确,它是以原子时为基础,在时刻上尽量接近世界时的一种时间计量系统。它的出现是现代社会对于精确计时的需要。

  

四丶夏令时

 夏令时,表示为了节约能源,人为规定时间的意思。也叫夏时制,夏时令(Daylight Saving Time:DST),又称“日光节约时制”和“夏令时间”,在这一制度实行期间所采用的统一时间称为“夏令时间”。一般在天亮早的夏季人为将时间调快一小时,可以使人早起早睡,减少照明量,以充分利用光照资源,从而节约照明用电。各个采纳夏时制的国家具体规定不同。目前全世界有近110个国家每年要实行夏令时。

  中国已不使用夏令时.

 

五丶区时转换

  计算区时(时间) - 已知区时(时间) = 计算时区 - 已知区时的时区.

  所以, 计算区时(时间)= 已知区时(时间) + 计算时区 - 已知区时的时区.

  需要注意的是, 有的地区在夏季使用夏令时, 即时钟会拨快一个小时.

 

  DateUtils.java

 /**
     * 将本地时间, 转换成目标时区的时间
     * @param sourceDate
     * @param targetZoneId {@link ZoneIds}
     * @return
     */
    public static Date convertTimezone(Date sourceDate, String targetZoneId){
        return convertTimezone(sourceDate, TimeZone.getTimeZone(targetZoneId));
    }

    public static Date convertTimezone(Date sourceDate, String sourceZoneId, String targetZoneId){
        TimeZone sourceTimeZone=TimeZone.getTimeZone(sourceZoneId);
        TimeZone targetTimeZone=TimeZone.getTimeZone(targetZoneId);

        return convertTimezone(sourceDate, sourceTimeZone, targetTimeZone);
    }

    /**
     * 将本地时间,转换成对应时区的时间
     * @param localDate
     * @param targetTimezone 转换成目标时区所在的时间
     * @return
     */
    public static Date convertTimezone(Date localDate, TimeZone targetTimezone){
        return convertTimezone(localDate, TimeZone.getDefault(), targetTimezone);
    }


    /**
     * 将sourceDate转换成指定时区的时间
     * @param sourceDate
     * @param sourceTimezone sourceDate所在的时区
     * @param targetTimezone 转化成目标时间所在的时区
     * @return
     */
    public static Date convertTimezone(Date sourceDate, TimeZone sourceTimezone, TimeZone targetTimezone){


        // targetDate - sourceDate=targetTimezone-sourceTimezone
        // --->
        // targetDate=sourceDate + (targetTimezone-sourceTimezone)


        Calendar calendar=Calendar.getInstance();
// date.getTime() 为时间戳, 为格林尼治到系统现在的时间差,世界各个地方获取的时间戳是一样的,
// 格式化输出时,因为设置了不同的时区,所以输出不一样
long sourceTime=sourceDate.getTime(); calendar.setTimeZone(sourceTimezone); calendar.setTimeInMillis(sourceTime);// 设置之后,calendar会计算各种filed对应的值,并保存 //获取源时区的到UTC的时区差 int sourceZoneOffset=calendar.get(Calendar.ZONE_OFFSET); calendar.setTimeZone(targetTimezone); calendar.setTimeInMillis(sourceTime); int targetZoneOffset=calendar.get(Calendar.ZONE_OFFSET); int targetDaylightOffset=calendar.get(Calendar.DST_OFFSET); // 夏令时 long targetTime=sourceTime+ (targetZoneOffset+targetDaylightOffset) -sourceZoneOffset; return new Date(targetTime); }

 

  ZoneIds.java

/**
 *
 * @see java.time.ZoneId#SHORT_IDS
 * @author TimFruit
 * @date 19-11-2 下午6:02
 */
public class ZoneIds {

    /*
    EST - -05:00
HST - -10:00
MST - -07:00
ACT - Australia/Darwin
AET - Australia/Sydney
AGT - America/Argentina/Buenos_Aires
ART - Africa/Cairo
AST - America/Anchorage
BET - America/Sao_Paulo
BST - Asia/Dhaka
CAT - Africa/Harare
CNT - America/St_Johns
CST - America/Chicago
CTT - Asia/Shanghai
EAT - Africa/Addis_Ababa
ECT - Europe/Paris
IET - America/Indiana/Indianapolis
IST - Asia/Kolkata
JST - Asia/Tokyo
MIT - Pacific/Apia
NET - Asia/Yerevan
NST - Pacific/Auckland
PLT - Asia/Karachi
PNT - America/Phoenix
PRT - America/Puerto_Rico
PST - America/Los_Angeles
SST - Pacific/Guadalcanal
VST - Asia/Ho_Chi_Min
     */

    public static  final String UTC="Z";// utc国际时间

    public static final String DEFAULT=TimeZone.getDefault().toZoneId().getId();

    public static final String BEIJING="Asia/Shanghai";  //也可以使用"+8" 北京在东8区




    //
//    UTC+10 夏莫罗标准时区
//    UTC-11 美属萨摩亚标准时区
//    UTC-10HST夏威夷-阿留申标准时区
//    UTC-9AKST阿拉斯加标准时区
//    UTC-8PST太平洋标准时区
//    UTC-7MST山地标准时区
//    UTC-6CST中部标准时区
//    UTC-5EST东部标准时区
//    UTC-4AST大西洋标准时区

    //https://baike.baidu.com/item/%E7%BE%8E%E5%9B%BD%E6%97%B6%E9%97%B4/3163209?fr=aladdin
    /*
    太平洋时区:代表城市洛杉矶,与北京相差16小时;
山地时区:代表城市盐湖城,与北京相差15小时;
中部时区:代表城市芝加哥,与北京相差14小时;
东部时区:代表城市纽约、华盛顿,与北京相差13小时;
夏威夷时区:代表城市:火奴鲁鲁,与北京相差18小时;
阿拉斯加时区:代表城市:费尔班克斯,与北京相差17小时。
     */
    public static final String US_EST="-5";  //东部标准时区
    public static  final String US_CST="-6";// 中部标准时区
    public static  final String US_MST="-7";// 山地标准时区
    public static final String US_PST="America/Los_Angeles";  //也可以使用"-8" 太平洋标准时区


    public static final String JST="Asia/Tokyo";//日本东京


}

 

  测试:

@Test
    public void convertTimezonePstNowTest(){
        // 太平洋时区:代表城市洛杉矶,与北京相差16小时;
        // 但由于实行夏令时, 夏季会快一个小时
        Date now=new Date();

        convertTimeZonePstTest(now);

    }


    @Test
    public void convertTimezonePstTest1(){
        // 太平洋时区:代表城市洛杉矶,与北京相差16小时;
        // 但由于实行夏令时, 夏季会快一个小时
        Date now=DateUtils.parse("2019-11-03 03:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 06:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 09:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 11:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 14:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 16:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 18:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 19:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 20:00:00");
        convertTimeZonePstTest(now);

        now=DateUtils.parse("2019-11-03 23:00:00");
        convertTimeZonePstTest(now);
    }


    private void convertTimeZonePstTest(Date sourceDate){
        Date target=DateUtils.convertTimezone(sourceDate, ZoneIds.US_PST);
        long sub=sourceDate.getTime()-target.getTime();
        System.out.println("北京时间与洛杉矶时间相差时间: "+sub/(60*60*1000) +" 小时");


        //由于有夏令时, 使用jdk提供的方法验证
        SimpleDateFormat pstSdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        pstSdf.setTimeZone(TimeZone.getTimeZone(ZoneIds.US_PST));
        String expectFormat=pstSdf.format(sourceDate);

        String targetFormat=DateUtils.format(target);
        System.out.println("洛杉矶时间: "+targetFormat);
        Assert.assertEquals(expectFormat,targetFormat);
    }

 

 

 

 

 

完整源码

 

posted @ 2019-11-03 22:29  timfruit  阅读(10557)  评论(1编辑  收藏  举报