2014-12-05 105 views
5

我错过了什么? Date()是自1970年1月1日午夜以来经过的毫秒数。如果午夜不是从凌晨0点开始?Java时间:从1970年1月1日凌晨1点开始?

参考文献:

我的测试程序:

package be.test.package.time; 

import java.text.DateFormat; 
import java.text.ParseException; 
import java.text.SimpleDateFormat; 
import java.util.ArrayList; 
import java.util.Date; 
import java.util.List; 
import java.util.Locale; 

public class TimeWork { 

    public static void main(String[] args) {  

     List<Long> longs = new ArrayList<>(); 
     List<String> strings = new ArrayList<>(); 

     DateFormat formatter = new SimpleDateFormat("dd-MM-yyyy hh:mm:ss.SSS"); 

     //Now 
     Date now = new Date(); 
     strings.add(formatter.format(now)); 

     //Test dates 
     strings.add("01-01-1970 00:00:00.000"); 
     strings.add("01-01-1970 01:00:00.000"); 
     strings.add("31-11-1969 00:00:00.000"); 
     strings.add("01-01-2014 00:00:00.000"); 

     //Test data 
     longs.add(-1L); 
     longs.add(0L); 
     longs.add(1L); 
     longs.add(7260000L); 
     longs.add(1417706084037L); 
     longs.add(-7260000L); 

     //Show the long value of the date 
     for (String string: strings) { 
      try { 
       Date date = formatter.parse(string); 
       System.out.println("Formated date : " + string + " = Long = " + date.getTime()); 
      } catch (ParseException e) { 
       e.printStackTrace(); 
      } 
     } 

     //Show the date behind the long 
     for (Long lo : longs) { 
      Date date = new Date(lo); 
      String string = formatter.format(date); 
      System.out.println("Formated date : " + string + " = Long = " + lo);   
     } 
    } 
} 

这是结果:

Formated date : 05-12-2014 08:54:59.318 = Long = 1417766099318 
Formated date : 01-01-1970 00:00:00.000 = Long = -3600000 
Formated date : 01-01-1970 01:00:00.000 = Long = 0 
Formated date : 31-11-1969 00:00:00.000 = Long = -2682000000 
Formated date : 01-01-2014 00:00:00.000 = Long = 1388530800000 
Formated date : 01-01-1970 12:59:59.999 = Long = -1 
Formated date : 01-01-1970 01:00:00.000 = Long = 0 
Formated date : 01-01-1970 01:00:00.001 = Long = 1 
Formated date : 01-01-1970 03:01:00.000 = Long = 7260000 
Formated date : 04-12-2014 04:14:44.037 = Long = 1417706084037 
Formated date : 31-12-1969 10:59:00.000 = Long = -7260000 

为什么:

Formated date : 01-01-1970 01:00:00.000 = Long = 0 

这是凌晨1点。我期待着凌晨0点。

+8

你是什么时区的java.time? – 2014-12-05 08:00:22

+1

我的时区是CET - 布鲁塞尔/比利时时间。 – 2014-12-05 08:01:47

+2

你领先于你自己的时代:-) – 2014-12-05 08:03:37

回答

8

1970年1月1日午夜UTC。您需要TimeZone

public static void main(String[] args) { 
    TimeZone tz = TimeZone.getTimeZone("UTC"); 
    Calendar cal = Calendar.getInstance(tz); 
    cal.setTimeInMillis(0); 
    DateFormat sdf = new SimpleDateFormat("dd-MM-yyyy hh:mm:ss.SSS"); 
    sdf.setTimeZone(tz); 
    System.out.println(sdf.format(cal.getTime())); 
} 

输出是

01-01-1970 12:00:00.000 
+0

当我没有在'Calendar'实例中设置TimeZone时,我得到了相同的结果。但这是否意味着'DateFormat'取决于时区? – GameDroids 2014-12-05 08:08:52

+0

@GameDroids请记住,自从1970年1月1日@ 12:00.000以来,“Calendar”是一个奇特的结构。是的,'DateFormat'包含'TimeZone',你可能需要知道什么时候是什么时候,你是什么时候到了我下午3点。 – 2014-12-05 08:11:57

+1

感谢您的解释!我检查日历字符串时,它被设置为'HOUR_OF_DAY = 1',所以'DateFormat'必须已经将它转换回'UTC'时间。 – GameDroids 2014-12-05 08:16:03

3

由长0所代表的时间是午夜1/1/1970 UTC。这恰好是凌晨1点,1/1/1970 CET。但是,由于您使用的是SimpleDateFormat其时区设置为CET,

  • 1/1/1970凌晨1点是,当被格式化的日期显示的时间;
  • 解析1/1/1970凌晨1点给你长0.1

如果你有一个SimpleDateFormat其时区设置为UTC,所以行为是完全不同的代表的时间。

如果您想使用除CET以外的时区,则可以使用DateFormatsetTimeZone方法。

1

另外两个答案是正确的。

顺便说一下,使用Joda-Time库或java.time package(受Joda-Time启发),这种日期时间工作更容易。这两个库都有具有明确分配时区的日期 - 时间类(与java.util.Date/.Calendar不同)。

Joda-Time和java.time都使用相同的epoch1970-01-01T00:00:00Z

在生成日期时间值的文本表示时,两者都使用ISO 8601标准格式。

Joda-Time

在乔达时代2.7。

DateTime epoch = new DateTime(0 , DateTimeZone.UTC); 

java.time

在Java 8更新45

ZonedDateTime epoch = ZonedDateTime.ofInstant(Instant.EPOCH , ZoneOffset.UTC) ; 
相关问题