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 = ???
回答:
簡潔な答え
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でも同じです
私は個人的にユーティリティクラスを使用しています(ただし、バックポート互換ではありません)。
/**
* 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()メソッドを使用して、インスタントをjdk 8の任意の日付APIに変換できます。
LocalDateTime date = LocalDateTime.ofInstant(timestamp, ZoneId.systemDefault());
import java.sql.Date
あなたのファイルに持っているならば、それは壊れます:常に投げるtoInstant()
方法java.sql.Date
。
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の実装は、私はOracleのJavaチュートリアル次の変換書き直し@に問題があったhttp://docs.oracle.com/javase/tutorial/datetime/iso/legacy.htmlを。
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
、Joda Timeではなく、JDK8 のクラスを使用するソリューションを探していることです。
LocalDate.from(Instant.ofEpochMilli(date.getTime()))
それはあなたのものと同等だと思いましたが、より直接的です。