java.util.Dateをjava.time.LocalDateに変換します


回答:


828

簡潔な答え

Date input = new Date();
LocalDate date = input.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();

説明

その名前にもかかわらjava.util.Dateず、「日付」ではなく、タイムライン上の瞬間を表します。オブジェクト内に保存される実際のデータは、long1970-01-01T00:00Z(1970 GMT / UTCの開始の真夜中)からのミリ秒数です。

java.util.DateJSR-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());

この新しい代替案はより直接的で、ガベージが少ないため、パフォーマンスが向上するはずです。


14
私はLocalDate.from(Instant.ofEpochMilli(date.getTime()))それはあなたのものと同等だと思いましたが、より直接的です。
Marko Topolnik、2014年

9
コンパイルするが実行しない@MarkoTopolnik。インスタントにはタイムゾーンが含まれていないため、LocalDateを取得する方法はありません。
JodaStephen 2014年

11
@assylias sqlDate.toLocalDate()を使用するだけです!
JodaStephen 2014年

25
@JodaStephenにDateはタイムゾーンの概念がなく、タイムゾーンInstantに関する情報も含まれていません。LocalDateAPIは、「タイムゾーンのない日」と言います。では、なぜニーズDateInstantに変換LocalDateするのatZone(ZoneId.systemDefault())でしょうか。
グスタボ、

8
@Gustavo:LocalDateまた、LocalDateTime「時刻またはタイムゾーンを保存または表す」ことはできません(参照:javadocs)それらはそれを格納しませんが、クラスはLocal日付や時刻を表すため、ローカルの日付/時刻への変換はタイムゾーンを意味します。
Cuga 2015

158

より良い方法は:

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ドライバーによってオーバーライドされる可能性がある場合)を使用します。それ以外の場合は、上記のメソッドを使用します。


12
これがより良い方法である場合、それは非常に醜くて冗長です。とても痛い。
ceklock 2015年

3
それは受け入れられた答えと比較して良いです、私は理由を説明します。はい、それは醜いですが、それが書いDateConvertUtilsた理由です。
Oliv 2015

新しいAPIに変換クラスを実装しなかった理由がわかりません。
ceklock 2017年

@ceklock、彼ら実装しました、変換クラスではなく、のようないくつかの変換メソッドDate.toInstant()
Ole VV 2017年

2
開発者が次のことができるように、これらを拡張関数としてパッケージ化するKotlinライブラリはありますか?日付x = ...; x.asLocalDate();
マークアシュワース2018

20
LocalDate localDate = LocalDate.parse( new SimpleDateFormat("yyyy-MM-dd").format(date) );

7
ここでは、SimpleDateFormatインスタンスは現在のスレッドに限定されています。それはされて使用スレッドセーフな方法で。現在、SimpleDateFormat(インスタンス化にコストがかかる)(必要なすべての内部データ構造のため)と言われていますが、実際にはスレッドはないため、(シングルアクセス)として共有することはできません(アクセスを同期しないと)。安全。(これでコードを「汚染」するコードがスレッドのライフサイクルの原因であった場合、ソリューションは機能しますが、それはめったに起こりません)。ぎこちない。を使用する理由は回避です。ThreadLocalThreadSimpleDateFormatjavax.time
デビッドブロック

6
このアプローチには多くのオーバーヘッドがあります。「高価」SimpleDateFormat(破棄される)、中間文字列(破棄される)、および解析のコストです。これ解決策ですが、お勧めできません。
David Bullock

2
@ceklockが、スレッドセーフではなくなるという問題が発生する
flup

4
@ceklock SimpleDateFormatは、一度に1つの日付しか処理できません。同時に使用すると、壊れた結果が生成されます。はい、それは重要です。SimpleDateFormatの単一インスタンスをグローバル変数に作成しないでください。
16

19

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));

4
真実ではありません。@ JodaStephenの答えはまだ機能します。java.util.DateをInstantに変換する別の方法が必要です。:このためにはorg.threeten.bp.DateTimeUtils.toInstant使用することができますthreeten.org/threetenbp/apidocs/org/threeten/bp/...
クリスチャンCiach

3
私がバックポートを使用していたときは、DateTimeUtilsは使用できませんでしたが、ThreeTenバックポート1.0以降を使用しているすべてのユーザーが使用できます。指摘してくれてありがとう。
dhalsim2 2016年

14
LocalDate ld = new java.sql.Date( new java.util.Date().getTime() ).toLocalDate();

1
そして最も簡単な実現:LocalDate.of(getYear()+ 1900、getMonth()+ 1、getDate())
Grigory Kislin

非推奨と言う行:|
TheRealChx101

8

あなたは一行で変換することができます:

public static LocalDate getLocalDateFromDate(Date date){
   return LocalDate.from(Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()));
}

java.time.DateTimeException:Unable to get LocalDate from TemporalAccessor:2018-01-31T11:54:27.964Z of type java.time.Instant
Luigi Rubino

@LuigiRubino指摘してくれてありがとう。更新された回答をご覧ください。以前にゾーンを追加するのを忘れていました。
Sahil Chhabra 2018年

8

まず、日付をインスタントに変換するのは簡単です

Instant timestamp = new Date().toInstant(); 

次に、ofInstant()メソッドを使用して、インスタントをjdk 8の任意の日付APIに変換できます。

LocalDateTime date = LocalDateTime.ofInstant(timestamp, ZoneId.systemDefault()); 

localDate、localDateTime、localTimeが含まれます-Shedom
Wei

ここでZoneIdを指定するとはどういう意味ですか。APIから日付->インスタントを取得し、UTCで1970年からのミリ秒と考えることができます。どのタイムゾーンにも変換されないAPIの観点から、対応するLocalDate(yyyy-mm-dd)のみを取得する場合、ZoneOffset.UTCを使用して、ローカルタイムゾーンへのこのインスタントオフセットを設定しないでください。ZoneId.systemDefault()の例は、サーバーから日付をシフトすることではありませんか?例 インスタントは2018-10-20 0:00を表し、UTCからAmerica / Los_Angelesにシフトします。ローカル日付2018-10-19では、2018-10-20ではなくですか?
のMichałZiobro

あなたがたまたまimport java.sql.Dateあなたのファイルに持っているならば、それは壊れます:常に投げるtoInstant()方法java.sql.Date
Oliv

4
Date input = new Date();
LocalDateTime  conv=LocalDateTime.ofInstant(input.toInstant(), ZoneId.systemDefault());
LocalDate convDate=conv.toLocalDate();

Dateインスタンスは、一方で、日付とあまり一緒に時間を含んでいLocalDateません。したがって、最初にそれLocalDateTimeをそのメソッドを使用するように変換し、次に時間がofInstant()ない場合はインスタンスをに変換できます。LocalDate


1
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();
    }

0

私は、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();

-2

この1つの単純な行の何が問題になっていますか?

new LocalDateTime(new Date().getTime()).toLocalDate();

Javaは、プリミティブ型longでtoLocalDate()を呼び出すことができないと不平を言います。実際のDateオブジェクトを使用しました。おそらく摩擦がありますか?
TheGeeko61 2018年

-8

私は以下の解決策でこの質問を解決しました

  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」の形式で日付を出力します


1
問題はjava.time、Joda Timeではなく、JDK8 のクラスを使用するソリューションを探していることです。
pioto 16
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.