All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.time4j.engine.Calendrical Maven / Gradle / Ivy

There is a newer version: 4.38
Show newest version
/*
 * -----------------------------------------------------------------------
 * Copyright © 2013-2015 Meno Hochschild, 
 * -----------------------------------------------------------------------
 * This file (Calendrical.java) is part of project Time4J.
 *
 * Time4J is free software: You can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as published
 * by the Free Software Foundation, either version 2.1 of the License, or
 * (at your option) any later version.
 *
 * Time4J is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with Time4J. If not, see .
 * -----------------------------------------------------------------------
 */

package net.time4j.engine;


/**
 * 

Abstract base class of all plain calendar date types which are * convertible via their day epoch numbers.

* * @param generic type of time unit compatible to {@link ChronoUnit}) * @param generic type of self reference * @author Meno Hochschild * @serial exclude */ /*[deutsch] *

Abstrakte Basisklasse aller reinen Datumstypen, die über ihre * Epochentage ineinander konvertierbar sind.

* * @param generic type of time unit compatible to {@link ChronoUnit}) * @param generic type of self reference * @author Meno Hochschild * @serial exclude */ public abstract class Calendrical> extends TimePoint implements CalendarDate { //~ Methoden ---------------------------------------------------------- /** *

Converts this calendar date to the given target type based on * the count of days relative to UTC epoch [1972-01-01].

* *

The conversion occurs on the local timeline at noon. This * reference time ensures that all date types remain convertible * even if a calendar system defines dates not starting at midnight.

* * @param generic target date type * @param target chronological type this date shall be converted to * @return converted date of target type T * @throws IllegalArgumentException if the target class does not * have any chronology * @throws ArithmeticException in case of numerical overflow */ /*[deutsch] *

Konvertiert dieses Datum zum angegebenen Zieltyp auf Basis der * Anzahl der Tage relativ zur UTC-Epoche [1972-01-01].

* *

Die Konversion findet auf dem lokalen Zeitstrahl um 12 Uhr mittags * als angenommener Referenzzeit statt. Diese Referenzzeit stellt sicher, * daß alle Datumstypen konvertierbar bleiben, auch wenn in einem * Kalendersystem ein Tag nicht um Mitternacht startet.

* * @param generic target date type * @param target chronological type this date shall be converted to * @return converted date of target type T * @throws IllegalArgumentException if the target class does not * have any chronology * @throws ArithmeticException in case of numerical overflow */ public > T transform(Class target) { String ref = target.getName(); Chronology chronology = Chronology.lookup(target); if (chronology == null) { // kommt normal nie vor, weil sich jede Chrono selbst registriert throw new IllegalArgumentException( "Cannot find any chronology for given target type: " + ref); } return this.transform(chronology.getCalendarSystem(), ref); } /** *

Converts this calendar date to the given target type based on * the count of days relative to UTC epoch [1972-01-01].

* *

The conversion occurs on the local timeline at noon. This * reference time ensures that all date types remain convertible * even if a calendar system defines dates not starting at midnight.

* * @param generic target date type * @param target chronological type this date shall be converted to * @param variant desired calendar variant * @return converted date of target type T * @throws IllegalArgumentException if the target class does not have any chronology * @throws ArithmeticException in case of numerical overflow * @since 3.5/4.3 */ /*[deutsch] *

Konvertiert dieses Datum zum angegebenen Zieltyp auf Basis der * Anzahl der Tage relativ zur UTC-Epoche [1972-01-01].

* *

Die Konversion findet auf dem lokalen Zeitstrahl um 12 Uhr mittags * als angenommener Referenzzeit statt. Diese Referenzzeit stellt sicher, * daß alle Datumstypen konvertierbar bleiben, auch wenn in einem * Kalendersystem ein Tag nicht um Mitternacht startet.

* * @param generic target date type * @param target chronological type this date shall be converted to * @param variant desired calendar variant * @return converted date of target type T * @throws IllegalArgumentException if the target class does not have any chronology * @throws ArithmeticException in case of numerical overflow * @since 3.5/4.3 */ public > T transform( Class target, String variant ) { String ref = target.getName(); Chronology chronology = Chronology.lookup(target); if (chronology == null) { // kommt normal nie vor, weil sich jede Chrono selbst registriert throw new IllegalArgumentException( "Cannot find any chronology for given target type: " + ref); } return this.transform(chronology.getCalendarSystem(variant), ref); } /** *

Converts this calendar date to the given target type based on * the count of days relative to UTC epoch [1972-01-01].

* *

The conversion occurs on the local timeline at noon. This * reference time ensures that all date types remain convertible * even if a calendar system defines dates not starting at midnight.

* * @param generic target date type * @param target chronological type this date shall be converted to * @param variantSource source of desired calendar variant * @return converted date of target type T * @throws ChronoException if given variant is not recognized * @throws IllegalArgumentException if the target class does not have any chronology * @throws ArithmeticException in case of numerical overflow * @since 3.6/4.4 */ /*[deutsch] *

Konvertiert dieses Datum zum angegebenen Zieltyp auf Basis der * Anzahl der Tage relativ zur UTC-Epoche [1972-01-01].

* *

Die Konversion findet auf dem lokalen Zeitstrahl um 12 Uhr mittags * als angenommener Referenzzeit statt. Diese Referenzzeit stellt sicher, * daß alle Datumstypen konvertierbar bleiben, auch wenn in einem * Kalendersystem ein Tag nicht um Mitternacht startet.

* * @param generic target date type * @param target chronological type this date shall be converted to * @param variantSource source of desired calendar variant * @return converted date of target type T * @throws ChronoException if given variant is not recognized * @throws IllegalArgumentException if the target class does not have any chronology * @throws ArithmeticException in case of numerical overflow * @since 3.6/4.4 */ public > T transform( Class target, VariantSource variantSource ) { return this.transform(target, variantSource.getVariant()); } @Override public boolean isBefore(CalendarDate date) { return (this.compareByTime(date) < 0); } @Override public boolean isAfter(CalendarDate date) { return (this.compareByTime(date) > 0); } @Override public boolean isSimultaneous(CalendarDate date) { return ((this == date) || (this.compareByTime(date) == 0)); } /** *

Defines a total respective natural order.

* *

This implementation first evaluates the temporal position on the * common timeline, that is the epoch day numbers. Only date objects * of the same calendrical type are comparable. The order is consistent * with {@code equals()} as long as subclasses don't define further * state attributes. If objects of different calendrical type are to be * compared on the timeline only applications can either use an * {@code EpochDays}-instance as {@code Comparator} or use one of * the {@code Temporal}-methods {@code isAfter()}, {@code isBefore()} * and {@code isSimultaneous()}.

* * @throws ClassCastException if there are different date types * @see EpochDays#compare(ChronoDisplay, ChronoDisplay) * @see #isBefore(CalendarDate) * @see #isAfter(CalendarDate) */ /*[deutsch] *

Definiert eine totale respektive eine natürliche Ordnung.

* *

Diese Implementierung wertet die zeitliche Position auf dem * gemeinsamen Zeitstrahl aus, also die Epochentage. Nur Datumsobjekte * gleichen Kalendertyps können miteinander verglichen werden. * Die Sortierung ist daher konsistent mit {@code equals()}, solange * Subklassen nicht weitere Zustandsattribute definieren. Sollen garantiert * Datumsobjekte verschiedenen Typs nur zeitlich verglichen werden, kann * entweder eine {@code EpochDays}-Instanz als {@code Comparator} * oder eine der {@code Temporal}-Methoden {@code isAfter()}, * {@code isBefore()} und {@code isSimultaneous()} verwendet werden.

* * @throws ClassCastException if there are different date types * @see EpochDays#compare(ChronoDisplay, ChronoDisplay) * @see #isBefore(CalendarDate) * @see #isAfter(CalendarDate) */ @Override public int compareTo(D date) { Class t1 = this.getChronology().getChronoType(); Class t2 = date.getChronology().getChronoType(); if (t1 != t2) { throw new ClassCastException( "Cannot compare different types of dates, " + "use instance of EpochDays as comparator instead."); } return this.compareByTime(date); } /** *

Based on the epoch day number and the calendar system.

* *

In other words: Two date object are equal if they have the * same temporal position on the local timeline and have the same * calendrical type. Subclasses which define further state attributes * must override this method.

* *

If an only temporal comparison is required then the method * {@link #isSimultaneous(CalendarDate)} is to be used.

* * @see Chronology#getChronoType() */ /*[deutsch] *

Basiert auf den Epochentagen und dem Kalendersystem.

* *

Mit anderen Worten: Zwei Datumsobjekte sind genau dann gleich, wenn * sie zeitlich gleich UND vom selben Kalendertyp sind. Subklassen, die * weitere Zustandsattribute definieren, müssen diese Methode * geeignet überschreiben.

* *

Soll ein rein zeitlicher Vergleich sichergestellt sein, dann * ist stattdessen die Methode {@link #isSimultaneous(CalendarDate)} * zu verwenden.

* * @see Chronology#getChronoType() */ @Override public boolean equals(Object obj) { if (this == obj) { return true; } else if (obj == null) { return false; } else if (obj instanceof Calendrical) { Calendrical that = (Calendrical) obj; Class t1 = this.getChronology().getChronoType(); Class t2 = that.getChronology().getChronoType(); return ( (t1 == t2) && (this.getDaysSinceEpochUTC() == that.getDaysSinceEpochUTC()) ); } else { return false; } } /** *

Based on the epoch day number.

*/ /*[deutsch] *

Basiert auf den Epochentagen.

*/ @Override public int hashCode() { long days = this.getDaysSinceEpochUTC(); return (int) (days ^ (days >>> 32)); } /** *

Adds given calendar days to this instance.

* * @param days calendar days to be added * @return result of addition * @throws ArithmeticException in case of numerical overflow * @since 3.4/4.3 */ /*[deutsch] *

Addiert die angegebenen Kalendertage zu dieser Instanz.

* * @param days calendar days to be added * @return result of addition * @throws ArithmeticException in case of numerical overflow * @since 3.4/4.3 */ public D plus(CalendarDays days) { long result = Math.addExact(this.getDaysSinceEpochUTC(), days.getAmount()); try { return this.getCalendarSystem().transform(result); } catch (IllegalArgumentException iae) { ArithmeticException ex = new ArithmeticException("Out of range: " + result); ex.initCause(iae); throw ex; } } /** *

Subtracts given calendar days from this instance.

* * @param days calendar days to be subtracted * @return result of subtraction * @throws ArithmeticException in case of numerical overflow * @since 3.4/4.3 */ /*[deutsch] *

Subtrahiert die angegebenen Kalendertage von dieser Instanz.

* * @param days calendar days to be subtracted * @return result of subtraction * @throws ArithmeticException in case of numerical overflow * @since 3.4/4.3 */ public D minus(CalendarDays days) { return this.plus(CalendarDays.of(Math.negateExact(days.getAmount()))); } @Override public long getDaysSinceEpochUTC() { return this.getCalendarSystem().transform(this.getContext()); } /** *

Definiert eine rein zeitliche Ordnung.

* *

Diese Implementierung wertet die zeitliche Position auf dem * gemeinsamen Zeitstrahl aus, also die Epochentage.

* * @param date another date to be compared with * @return negative, zero or positive integer if this instance is earlier, * simultaneous or later than given date */ protected int compareByTime(CalendarDate date) { long d1 = this.getDaysSinceEpochUTC(); long d2 = date.getDaysSinceEpochUTC(); return ((d1 < d2) ? -1 : ((d1 == d2) ? 0 : 1)); } private CalendarSystem getCalendarSystem() { return this.getChronology().getCalendarSystem(); } private T transform( CalendarSystem calsys, String ref ) { long utcDays = this.getDaysSinceEpochUTC(); if ( (calsys.getMinimumSinceUTC() > utcDays) || (calsys.getMaximumSinceUTC() < utcDays) ) { throw new ArithmeticException("Cannot transform <" + utcDays + "> to: " + ref); } else { return calsys.transform(utcDays); } } }