Java:为什么Date构造函数不推荐,我用什么来代替?
我来自C#世界,所以对Java还没有太多的经验。 Eclipse刚刚告知, Date
已被弃用。
Person p = new Person(); p.setDateOfBirth(new Date(1985, 1, 1));
为什么? 什么(特别是在上面的情况下)应该用来代替?
具体的date构造函数已被弃用,应该使用日历。 JavaDoc for Date描述了哪些构造函数被弃用,以及如何使用日历来replace它们。
java.util.Date
类实际上并不被弃用,只是构造函数和其他一些构造函数/方法都被弃用了。 它被弃用,因为这种使用不符合国际化。 应使用Calendar
类代替:
Calendar cal = Calendar.getInstance(); cal.set(Calendar.YEAR, 1988); cal.set(Calendar.MONTH, Calendar.JANUARY); cal.set(Calendar.DAY_OF_MONTH, 1); Date dateRepresentation = cal.getTime();
看看dateJavadoc:
http://download.oracle.com/javase/6/docs/api/java/util/Date.html
当Java第一次启动和发展时, java.util.Date
, java.util.Calendar
和java.text.SimpleDateFormat
类被冲得太快了。 这些课程没有很好的devise或实施。 试图改进,因此你已经发现的弃用。 不幸的是,改进的尝试很大程度上失败 你应该完全避免这些类 。 它们在Java 8中被新类所取代。
在你的代码中的问题
java.util.Date具有date和时间部分。 您忽略了代码中的时间部分。 因此,Date类将按照JVM的默认时区定义的那一天开始,并将该时间应用于Date对象。 所以你的代码的结果将取决于它运行的是哪台机器或者哪个时区被设置。 可能不是你想要的。
如果您只想要date,而不需要时间部分,例如出生date,则可能不想使用Date
对象。 您可能只想以YYYY-MM-DD
ISO 8601格式存储date的string。 或者使用Joda-Time中的LocalDate
对象(见下文)。
乔达时间
在Java中学习的第一件事: 避免与Java捆绑在一起的臭名昭着的java.util.Date&java.util.Calendar类 。
正如在user3277382的答案中正确指出的那样 ,在Java 8中使用Joda-Time或新的java.time。*包 。
Joda-Time中的示例代码2.3
DateTimeZone timeZoneNorway = DateTimeZone.forID( "Europe/Oslo" ); DateTime birthDateTime_InNorway = new DateTime( 1985, 1, 1, 3, 2, 1, timeZoneNorway ); DateTimeZone timeZoneNewYork = DateTimeZone.forID( "America/New_York" ); DateTime birthDateTime_InNewYork = birthDateTime_InNorway.toDateTime( timeZoneNewYork ); DateTime birthDateTime_UtcGmt = birthDateTime_InNorway.toDateTime( DateTimeZone.UTC ); LocalDate birthDate = new LocalDate( 1985, 1, 1 );
转储到控制台…
System.out.println( "birthDateTime_InNorway: " + birthDateTime_InNorway ); System.out.println( "birthDateTime_InNewYork: " + birthDateTime_InNewYork ); System.out.println( "birthDateTime_UtcGmt: " + birthDateTime_UtcGmt ); System.out.println( "birthDate: " + birthDate );
当运行…
birthDateTime_InNorway: 1985-01-01T03:02:01.000+01:00 birthDateTime_InNewYork: 1984-12-31T21:02:01.000-05:00 birthDateTime_UtcGmt: 1985-01-01T02:02:01.000Z birthDate: 1985-01-01
java.time
在这种情况下, java.time的代码与Joda-Time的代码几乎相同。
我们得到一个时区( ZoneId
),并构造一个分配给该时区的date – 时间对象( ZonedDateTime
)。 然后,使用不可变对象模式,我们基于旧对象的同一时刻(自时代以来的纳秒数)创build新的date时间,但分配了其他时区。 最后我们得到一个没有时间和时区的LocalDate
,但是在确定这个date的时候会注意到这个时区的适用情况(比如在奥斯陆的一个新的一天比在纽约早)。
ZoneId zoneId_Norway = ZoneId.of( "Europe/Oslo" ); ZonedDateTime zdt_Norway = ZonedDateTime.of( 1985 , 1 , 1 , 3 , 2 , 1 , 0 , zoneId_Norway ); ZoneId zoneId_NewYork = ZonedId.of( "America/New_York" ); ZonedDateTime zdt_NewYork = zdt_Norway.withZoneSameInstant( zoneId_NewYork ); ZonedDateTime zdt_Utc = zdt_Norway.withZoneSameInstant( ZoneOffset.UTC ); // Or, next line is similar. Instant instant = zdt_Norway.toInstant(); // Instant is always in UTC. LocalDate localDate_Norway = zdt_Norway.toLocalDate();
关于java.time
java.time框架内置于Java 8及更高版本中。 这些类取代了麻烦的旧的遗留date时间类,如java.util.Date
, Calendar
和SimpleDateFormat
。
Joda-Time项目现在处于维护模式 ,build议迁移到java.time类。
要了解更多信息,请参阅Oracle教程 。 并search堆栈溢出了很多例子和解释。 规范是JSR 310 。
从何处获取java.time类?
- Java SE 8 , Java SE 9和更高版本
- 内置。
- 带有捆绑实现的标准Java API的一部分。
- Java 9增加了一些小function和修复。
- Java SE 6和Java SE 7
- 大部分的java.timefunction都被移植到了ThreeTen-Backport中的 Java 6&7中。
- Android的
- ThreeTenABP项目专门针对Android,采用了ThreeTen-Backport (上文提到)。
- 请参阅如何使用ThreeTenABP …。
ThreeTen-Extra项目将java.time扩展到其他类。 这个项目是未来可能增加java.time的一个试验场。 你可能会在这里find一些有用的类,比如Interval
, YearWeek
, YearQuarter
等等 。
我把这个问题看作是一个新问题的重复,它问到在特定的年份,月份和date没有被弃用的方式来获取Date
。
到目前为止,这里的答案都是使用Calendar
类,直到Java 8推出之前,情况才是真实的。 但是从Java 8开始,标准的做法是:
LocalDate localDate = LocalDate.of(1985, 1, 1);
然后如果你确实需要一个java.util.Date
,你可以使用这个问题中的build议。
有关更多信息,请查看Java 8 的API或教程 。
请注意, date时间部分默认为当前时间,因此Calendar.getTime()
是不确定的。
要重现,请尝试运行以下代码几次:
Calendar c = Calendar.getInstance(); c.set(2010, 2, 7); // NB: 2 means March, not February! System.err.println(c.getTime());
输出例如:
Sun Mar 07 10:46:21 CET 2010
几分钟后运行完全相同的代码将产生:
Sun Mar 07 10:57:51 CET 2010
所以,虽然set()
强制相应的字段更正值,但会泄漏其他字段的系统时间。 (使用Sun jdk6&jdk7进行testing)