将java.util.Date
对象转换为新的JDK 8 / JSR-310 的最佳方法是什么java.time.LocalDate
?
Date input = new Date();
LocalDate date = ???
将java.util.Date
对象转换为新的JDK 8 / JSR-310 的最佳方法是什么java.time.LocalDate
?
Date input = new Date();
LocalDate date = ???
Answers:
简短答案
Date input = new Date();
LocalDate date = input.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
说明
尽管有名称,它java.util.Date
代表时间轴上的一个瞬间,而不是“日期”。存储在对象中的实际数据是long
自1970-01-01T00:00Z(1970 GMT / UTC开始的午夜)以来的毫秒数。
java.util.Date
JSR-310中的等效类是Instant
,因此有一种方便的方法toInstant()
可以提供转换:
Date input = new Date();
Instant instant = input.toInstant();
一个java.util.Date
实例没有时区的概念。如果调用toString()
,这可能看起来很奇怪java.util.Date
,因为toString
相对于时区。但是,该方法实际上在运行时使用了Java的默认时区来提供字符串。时区不是的实际状态的一部分java.util.Date
。
一个Instant
还没有关于时区的任何信息。因此,要将Instant
日期转换为本地日期,必须指定时区。这可能是默认区域- ZoneId.systemDefault()
或可能是您的应用程序控制的时区,例如用户首选项中的时区。使用该atZone()
方法来应用时区:
Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
一个ZonedDateTime
包含由本地日期和时间,时区,与GMT / UTC的偏移量的状态。因此,LocalDate
可以使用toLocalDate()
以下命令轻松提取日期--
Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
LocalDate date = zdt.toLocalDate();
Java 9答案
在Java SE 9中,添加了一个新方法,该方法稍微简化了此任务:
Date input = new Date();
LocalDate date = LocalDate.ofInstant(input.toInstant(), ZoneId.systemDefault());
这个新的替代方法更直接,产生更少的垃圾,因此应该表现得更好。
Date
没有时区的概念,Instant
也不包含有关时区的信息。该LocalDate
API说,“没有一个时区有个约会”。那么为什么要从Date
转变Instant
为LocalDate
需求atZone(ZoneId.systemDefault())
呢?
LocalDate
并且LocalDateTime
不要“存储或表示时间或时区”(参考:javadocs)。尽管它们不存储-类确实代表Local
日期和/或时间,因此转换为本地日期/时间意味着时区。
更好的方法是:
Date date = ...;
Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()).toLocalDate()
此版本的优点:
无论输入是的实例java.util.Date
还是其的子类,都可以工作java.sql.Date
(与@JodaStephen的方式不同)。这对于JDBC起源的数据很常见。java.sql.Date.toInstant()
总是抛出异常。
具有JSR-310反向端口的JDK8和JDK7相同
我个人使用实用程序类(但它与backport不兼容):
/**
* Utilities for conversion between the old and new JDK date types
* (between {@code java.util.Date} and {@code java.time.*}).
*
* <p>
* All methods are null-safe.
*/
public class DateConvertUtils {
/**
* Calls {@link #asLocalDate(Date, ZoneId)} with the system default time zone.
*/
public static LocalDate asLocalDate(java.util.Date date) {
return asLocalDate(date, ZoneId.systemDefault());
}
/**
* Creates {@link LocalDate} from {@code java.util.Date} or it's subclasses. Null-safe.
*/
public static LocalDate asLocalDate(java.util.Date date, ZoneId zone) {
if (date == null)
return null;
if (date instanceof java.sql.Date)
return ((java.sql.Date) date).toLocalDate();
else
return Instant.ofEpochMilli(date.getTime()).atZone(zone).toLocalDate();
}
/**
* Calls {@link #asLocalDateTime(Date, ZoneId)} with the system default time zone.
*/
public static LocalDateTime asLocalDateTime(java.util.Date date) {
return asLocalDateTime(date, ZoneId.systemDefault());
}
/**
* Creates {@link LocalDateTime} from {@code java.util.Date} or it's subclasses. Null-safe.
*/
public static LocalDateTime asLocalDateTime(java.util.Date date, ZoneId zone) {
if (date == null)
return null;
if (date instanceof java.sql.Timestamp)
return ((java.sql.Timestamp) date).toLocalDateTime();
else
return Instant.ofEpochMilli(date.getTime()).atZone(zone).toLocalDateTime();
}
/**
* Calls {@link #asUtilDate(Object, ZoneId)} with the system default time zone.
*/
public static java.util.Date asUtilDate(Object date) {
return asUtilDate(date, ZoneId.systemDefault());
}
/**
* Creates a {@link java.util.Date} from various date objects. Is null-safe. Currently supports:<ul>
* <li>{@link java.util.Date}
* <li>{@link java.sql.Date}
* <li>{@link java.sql.Timestamp}
* <li>{@link java.time.LocalDate}
* <li>{@link java.time.LocalDateTime}
* <li>{@link java.time.ZonedDateTime}
* <li>{@link java.time.Instant}
* </ul>
*
* @param zone Time zone, used only if the input object is LocalDate or LocalDateTime.
*
* @return {@link java.util.Date} (exactly this class, not a subclass, such as java.sql.Date)
*/
public static java.util.Date asUtilDate(Object date, ZoneId zone) {
if (date == null)
return null;
if (date instanceof java.sql.Date || date instanceof java.sql.Timestamp)
return new java.util.Date(((java.util.Date) date).getTime());
if (date instanceof java.util.Date)
return (java.util.Date) date;
if (date instanceof LocalDate)
return java.util.Date.from(((LocalDate) date).atStartOfDay(zone).toInstant());
if (date instanceof LocalDateTime)
return java.util.Date.from(((LocalDateTime) date).atZone(zone).toInstant());
if (date instanceof ZonedDateTime)
return java.util.Date.from(((ZonedDateTime) date).toInstant());
if (date instanceof Instant)
return java.util.Date.from((Instant) date);
throw new UnsupportedOperationException("Don't know hot to convert " + date.getClass().getName() + " to java.util.Date");
}
/**
* Creates an {@link Instant} from {@code java.util.Date} or it's subclasses. Null-safe.
*/
public static Instant asInstant(Date date) {
if (date == null)
return null;
else
return Instant.ofEpochMilli(date.getTime());
}
/**
* Calls {@link #asZonedDateTime(Date, ZoneId)} with the system default time zone.
*/
public static ZonedDateTime asZonedDateTime(Date date) {
return asZonedDateTime(date, ZoneId.systemDefault());
}
/**
* Creates {@link ZonedDateTime} from {@code java.util.Date} or it's subclasses. Null-safe.
*/
public static ZonedDateTime asZonedDateTime(Date date, ZoneId zone) {
if (date == null)
return null;
else
return asInstant(date).atZone(zone);
}
}
asLocalDate()
这里的方法是null安全的,toLocalDate()
如果输入为java.sql.Date
(使用JDBC驱动程序可以重写它,以避免时区问题或不必要的计算),则使用,否则使用上述方法。
DateConvertUtils
。
Date.toInstant()
。
LocalDate localDate = LocalDate.parse( new SimpleDateFormat("yyyy-MM-dd").format(date) );
SimpleDateFormat
实例仅限于当前线程。它是用于在一个线程安全的方式。现在,SimpleDateFormat
被称为“实例化的昂贵”(由于它需要所有内部数据结构),但是您不能将一个实例共享为“单个”(不同步访问),因为它确实不是线程-安全。(如果此代码中的“污染”代码负责线程的生命周期,则该ThreadLocal
解决方案可以工作,但是这种情况很少发生)。尴尬。避免是使用的原因。Thread
SimpleDateFormat
javax.time
SimpleDateFormat
(被丢弃),中间字符串(被丢弃)以及解析成本。这是一个解决方案,但不建议这样做。
如果您使用的是Java 8,@ JodaStephen的答案显然是最好的。但是,如果您使用的是JSR-310反向端口,那么很遗憾,您必须执行以下操作:
Date input = new Date();
Calendar cal = Calendar.getInstance();
cal.setTime(input);
LocalDate date = LocalDate.of(cal.get(Calendar.YEAR),
cal.get(Calendar.MONTH) + 1,
cal.get(Calendar.DAY_OF_MONTH));
LocalDate ld = new java.sql.Date( new java.util.Date().getTime() ).toLocalDate();
您可以在一行中进行转换:
public static LocalDate getLocalDateFromDate(Date date){
return LocalDate.from(Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()));
}
首先,将日期转换为即时数很容易
Instant timestamp = new Date().toInstant();
然后,您可以使用ofInstant()方法将Instant转换为jdk 8中的任何date api:
LocalDateTime date = LocalDateTime.ofInstant(timestamp, ZoneId.systemDefault());
import java.sql.Date
在文件中,它将中断:始终抛出的toInstant()
方法java.sql.Date
。
Date input = new Date();
LocalDateTime conv=LocalDateTime.ofInstant(input.toInstant(), ZoneId.systemDefault());
LocalDate convDate=conv.toLocalDate();
该Date
实例确实包含时间以及日期,而LocalDate
没有。因此,您可以首先LocalDateTime
使用它的方法将其转换为,ofInstant()
然后,如果您想在没有时间的情况下将其转换为LocalDate
。
public static LocalDate Date2LocalDate(Date date) {
return LocalDate.parse(date.toString(), DateTimeFormatter.ofPattern("EEE MMM dd HH:mm:ss zzz yyyy"))
这种格式是从 Date#tostring
public String toString() {
// "EEE MMM dd HH:mm:ss zzz yyyy";
BaseCalendar.Date date = normalize();
StringBuilder sb = new StringBuilder(28);
int index = date.getDayOfWeek();
if (index == BaseCalendar.SUNDAY) {
index = 8;
}
convertToAbbr(sb, wtb[index]).append(' '); // EEE
convertToAbbr(sb, wtb[date.getMonth() - 1 + 2 + 7]).append(' '); // MMM
CalendarUtils.sprintf0d(sb, date.getDayOfMonth(), 2).append(' '); // dd
CalendarUtils.sprintf0d(sb, date.getHours(), 2).append(':'); // HH
CalendarUtils.sprintf0d(sb, date.getMinutes(), 2).append(':'); // mm
CalendarUtils.sprintf0d(sb, date.getSeconds(), 2).append(' '); // ss
TimeZone zi = date.getZone();
if (zi != null) {
sb.append(zi.getDisplayName(date.isDaylightTime(), TimeZone.SHORT, Locale.US)); // zzz
} else {
sb.append("GMT");
}
sb.append(' ').append(date.getYear()); // yyyy
return sb.toString();
}
我在JBoss EAP 6上使用@JodaStephen实现时遇到了问题。因此,我按照http://docs.oracle.com/javase/tutorial/datetime/iso/legacy.html中 Oracle Java教程的要求重写了转换。
Date input = new Date();
GregorianCalendar gregorianCalendar = (GregorianCalendar) Calendar.getInstance();
gregorianCalendar.setTime(input);
ZonedDateTime zonedDateTime = gregorianCalendar.toZonedDateTime();
zonedDateTime.toLocalDate();
这1条简单的线有什么问题?
new LocalDateTime(new Date().getTime()).toLocalDate();
我用下面的解决方案解决了这个问题
import org.joda.time.LocalDate;
Date myDate = new Date();
LocalDate localDate = LocalDate.fromDateFields(myDate);
System.out.println("My date using Date" Nov 18 11:23:33 BRST 2016);
System.out.println("My date using joda.time LocalTime" 2016-11-18);
在这种情况下,localDate将以“ yyyy-MM-dd”格式打印日期
java.time
JDK8中的类而不是Joda Time 寻找解决方案。
LocalDate.from(Instant.ofEpochMilli(date.getTime()))
以为它等同于您的,但是更直接。