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

net.time4j.Moment Maven / Gradle / Ivy

/*
 * -----------------------------------------------------------------------
 * Copyright © 2013-2017 Meno Hochschild, 
 * -----------------------------------------------------------------------
 * This file (Moment.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;

import net.time4j.base.GregorianMath;
import net.time4j.base.TimeSource;
import net.time4j.base.UnixTime;
import net.time4j.engine.AttributeQuery;
import net.time4j.engine.BridgeChronology;
import net.time4j.engine.CalendarFamily;
import net.time4j.engine.CalendarVariant;
import net.time4j.engine.Calendrical;
import net.time4j.engine.ChronoDisplay;
import net.time4j.engine.ChronoElement;
import net.time4j.engine.ChronoEntity;
import net.time4j.engine.ChronoException;
import net.time4j.engine.ChronoMerger;
import net.time4j.engine.ChronoOperator;
import net.time4j.engine.Chronology;
import net.time4j.engine.Converter;
import net.time4j.engine.DisplayStyle;
import net.time4j.engine.ElementRule;
import net.time4j.engine.EpochDays;
import net.time4j.engine.FlagElement;
import net.time4j.engine.RealTime;
import net.time4j.engine.StartOfDay;
import net.time4j.engine.Temporal;
import net.time4j.engine.ThreetenAdapter;
import net.time4j.engine.TimeAxis;
import net.time4j.engine.TimeLine;
import net.time4j.engine.TimePoint;
import net.time4j.engine.UnitRule;
import net.time4j.format.Attributes;
import net.time4j.format.CalendarText;
import net.time4j.format.CalendarType;
import net.time4j.format.ChronoPattern;
import net.time4j.format.DisplayMode;
import net.time4j.format.Leniency;
import net.time4j.format.TemporalFormatter;
import net.time4j.scale.LeapSecondEvent;
import net.time4j.scale.LeapSeconds;
import net.time4j.scale.TimeScale;
import net.time4j.scale.UniversalTime;
import net.time4j.tz.OverlapResolver;
import net.time4j.tz.TZID;
import net.time4j.tz.Timezone;
import net.time4j.tz.TransitionStrategy;
import net.time4j.tz.ZonalOffset;

import java.io.DataInput;
import java.io.DataOutput;
import java.io.IOException;
import java.io.InvalidObjectException;
import java.io.ObjectInputStream;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.time.Instant;
import java.time.format.DateTimeFormatter;
import java.time.temporal.ChronoField;
import java.time.temporal.TemporalAccessor;
import java.util.Collections;
import java.util.EnumMap;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;

import static net.time4j.PlainTime.*;
import static net.time4j.SI.NANOSECONDS;
import static net.time4j.SI.SECONDS;
import static net.time4j.scale.TimeScale.*;


/**
 * 

Represents an instant/moment on the universal timeline with reference * to the timezone UTC (UTC+00:00 / Greenwich-meridian).

* *

The JDK-equivalent is traditionally the class {@code java.util.Date}. * In contrast to that old class this class stores the elapsed time not just * in millisecond but in nanosecond precision on 96-bit base.

* *

Following elements which are declared as constants are registered by * this class with access in UTC timezone:

* *
    *
  • {@link #POSIX_TIME}
  • *
  • {@link #FRACTION}
  • *
  • {@link #PRECISION}
  • *
* *

Furthermore, most local elements like {@code PlainTime.ISO_HOUR} etc. * registered in class {@code PlainTimestamp} or those defined in * {@link Weekmodel} are indirectly supported via the queries in the * interface {@link ZonalElement}. A {@code Moment} is also capable of * delivering the date- and time-values in a different timezone if the * method {@link #toZonalTimestamp(TZID)} is called. If zonal operators * are defined by any elements then manipulations of related data are * possible in any timezone.

* *

Time arithmetic

* *

The main time units are defined by {@link SI} (counting possible * UTC-leapseconds) and {@link TimeUnit}. Latter unit type can be used * if a better interoperability is needed for external APIs which ignore * leapseconds. Both kinds of time units can be used in the methods * {@code plus(long, unit)}, {@code minus(long, unit)} and * {@code until(Moment, unit)}.

* *

Time scales

* *

Following table illustrates how a time scale affects values and representations * before, during and after a leap second event. Two views exist, either interpreting * a moment as count of seconds elapsed since an epoch or printed in a scale-specific representation.

* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
What does happen around a leap second?
time scaleepoch
Moment.of(0, scale)
value space
getElapsedTime(scale)
lexical space
toString(scale)
POSIX1970-01-01T00:00:00.000Z1483228799
1483228799
1483228800
2016-12-31T23:59:59.000Z
2016-12-31T23:59:59.000Z
2017-01-01T00:00:00.000Z
UTC1972-01-01T00:00:00.000Z1420156825
1420156826
1420156827
2016-12-31T23:59:59.000Z
2016-12-31T23:59:60.000Z
2017-01-01T00:00:00.000Z
TAI1971-12-31T23:59:50.000Z *)1420156835
1420156836
1420156837
2017-01-01T00:00:35.000Z
2017-01-01T00:00:36.000Z
2017-01-01T00:00:37.000Z
GPS1980-01-06T00:00:00.000Z1167264016
1167264017
1167264018
2017-01-01T00:00:16.000Z
2017-01-01T00:00:17.000Z
2017-01-01T00:00:18.000Z
*

*) This value is only virtual * since TAI is really supported first at UTC epoch (10 seconds later).

*
* * @author Meno Hochschild * @doctags.concurrency {immutable} */ /*[deutsch] *

Repräsentiert einen Zeitpunkt auf der Weltzeitlinie mit Bezug * auf die UTC-Zeitzone (UTC+00:00 / Greenwich-Meridian).

* *

Im JDK heißt das Äquivalent {@code java.util.Date}. Diese * Klasse speichert im Gegensatz zum JDK die Epochenzeit nicht in Milli-, * sondern in Nanosekunden auf 96-Bit-Basis.

* *

Registriert sind folgende als Konstanten deklarierte Elemente mit * Zugriff in der UTC-Zeitzone:

* *
    *
  • {@link #POSIX_TIME}
  • *
  • {@link #FRACTION}
  • *
  • {@link #PRECISION}
  • *
* *

Darüberhinaus sind die meisten lokalen Elemente wie * {@code PlainTime.ISO_HOUR} usw., die in der Klasse {@code PlainTimestamp} * registriert sind oder jene definiert in {@link Weekmodel}, indirekt * unterstützt, wenn über die entsprechenden Abfragen im Interface * {@link ZonalElement} eine Zeitzonenreferenz angegeben wird. Ein * {@code Moment} kann auch die Datums- und Zeitwerte in einer beliebigen * Zeitzone ausgeben, wenn die Methode {@link #toZonalTimestamp(TZID)} * aufgerufen wird. Wenn Elemente zonale Operatoren definieren, dann sind * Manipulationen der zugehörigen Daten in einer beliebigen Zeitzone * möglich.

* *

Zeitarithmetik

* *

Als Zeiteinheiten kommen {@link SI} (mit Zählung von Schaltsekunden) * und {@link TimeUnit} in Betracht. Letztere Einheit kann verwendet werden, * wenn eine bessere Interoperabilität mit externen APIs notwendig ist, * die UTC-Schaltsekunden ignorieren. Beide Arten von Zeiteinheiten werden in * den Methoden {@code plus(long, unit)}, {@code minus(long, unit)} und * {@code until(Moment, unit)} verwendet.

* *

Zeitskalen

* *

Folgende Tabelle illustriert, wie eine Zeitskala Werte und Darstellungen * von Moment-Objekten vor, während und nach einer Schaltsekunde * beeinflußt. Zwei Ansichten gibt es, entweder wird ein Moment als * seit einer Epoche verstrichene Anzahl von Sekunden interpretiert, oder der Moment * wird in einer skalenspezifischen Darstellung ausgegeben.

* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Was passiert zu einer Schaltsekunde?
time scaleepoch
Moment.of(0, scale)
value space
getElapsedTime(scale)
lexical space
toString(scale)
POSIX1970-01-01T00:00:00.000Z1483228799
1483228799
1483228800
2016-12-31T23:59:59.000Z
2016-12-31T23:59:59.000Z
2017-01-01T00:00:00.000Z
UTC1972-01-01T00:00:00.000Z1420156825
1420156826
1420156827
2016-12-31T23:59:59.000Z
2016-12-31T23:59:60.000Z
2017-01-01T00:00:00.000Z
TAI1971-12-31T23:59:50.000Z *)1420156835
1420156836
1420156837
2017-01-01T00:00:35.000Z
2017-01-01T00:00:36.000Z
2017-01-01T00:00:37.000Z
GPS1980-01-06T00:00:00.000Z1167264016
1167264017
1167264018
2017-01-01T00:00:16.000Z
2017-01-01T00:00:17.000Z
2017-01-01T00:00:18.000Z
*

*) Dieser Wert ist nur virtuell, weil TAI erst ab * der UTC-Epoche unterstützt wird (10 Sekunden später).

*
* * @author Meno Hochschild * @doctags.concurrency {immutable} */ @CalendarType("iso8601") public final class Moment extends TimePoint implements UniversalTime, Temporal, ThreetenAdapter { //~ Statische Felder/Initialisierungen -------------------------------- private static final long UTC_GPS_DELTA = ((1980 - 1972) * 365 + 2 + 5) * 86400 + 9; private static final long POSIX_UTC_DELTA = 2 * 365 * 86400; private static final long POSIX_GPS_DELTA = POSIX_UTC_DELTA + UTC_GPS_DELTA - 9; // -9 => without leap seconds private static final int MIO = 1000000; private static final int MRD = 1000000000; private static final int POSITIVE_LEAP_MASK = 0x40000000; private static final long MIN_LIMIT; private static final long MAX_LIMIT; static { long mjdMin = GregorianMath.toMJD(GregorianMath.MIN_YEAR, 1, 1); long mjdMax = GregorianMath.toMJD(GregorianMath.MAX_YEAR, 12, 31); MIN_LIMIT = EpochDays.UNIX.transform( mjdMin, EpochDays.MODIFIED_JULIAN_DATE) * 86400; MAX_LIMIT = EpochDays.UNIX.transform( mjdMax, EpochDays.MODIFIED_JULIAN_DATE) * 86400 + 86399; } private static final Moment MIN = new Moment(MIN_LIMIT, 0, POSIX); private static final Moment MAX = new Moment(MAX_LIMIT, MRD - 1, POSIX); private static final Moment START_LS_CHECK = new Moment(86400 + POSIX_UTC_DELTA, 0, POSIX); private static final Set> HIGH_TIME_ELEMENTS; private static final Map, Integer> LOW_TIME_ELEMENTS; private static final Map UNIT_LENGTHS; static { Set> high = new HashSet<>(); high.add(ISO_HOUR); high.add(DIGITAL_HOUR_OF_DAY); high.add(DIGITAL_HOUR_OF_AMPM); high.add(CLOCK_HOUR_OF_DAY); high.add(CLOCK_HOUR_OF_AMPM); high.add(AM_PM_OF_DAY); high.add(MINUTE_OF_HOUR); high.add(MINUTE_OF_DAY); HIGH_TIME_ELEMENTS = Collections.unmodifiableSet(high); Map, Integer> low = new HashMap<>(); low.put(SECOND_OF_MINUTE, Integer.valueOf(1)); low.put(SECOND_OF_DAY, Integer.valueOf(1)); low.put(MILLI_OF_SECOND, Integer.valueOf(1000)); low.put(MILLI_OF_DAY, Integer.valueOf(1000)); low.put(MICRO_OF_SECOND, Integer.valueOf(MIO)); low.put(MICRO_OF_DAY, Integer.valueOf(MIO)); low.put(NANO_OF_SECOND, Integer.valueOf(MRD)); low.put(NANO_OF_DAY, Integer.valueOf(MRD)); LOW_TIME_ELEMENTS = Collections.unmodifiableMap(low); Map unitLengths = new EnumMap<>(TimeUnit.class); unitLengths.put(TimeUnit.DAYS, 86400.0); unitLengths.put(TimeUnit.HOURS, 3600.0); unitLengths.put(TimeUnit.MINUTES, 60.0); unitLengths.put(TimeUnit.SECONDS, 1.0); unitLengths.put(TimeUnit.MILLISECONDS, 0.001); unitLengths.put(TimeUnit.MICROSECONDS, 0.000001); unitLengths.put(TimeUnit.NANOSECONDS, 0.000000001); UNIT_LENGTHS = Collections.unmodifiableMap(unitLengths); } private static final TimeAxis ENGINE; static { TimeAxis.Builder builder = TimeAxis.Builder.setUp( TimeUnit.class, Moment.class, new Merger(), MIN, MAX); for (TimeUnit unit : TimeUnit.values()) { builder.appendUnit( unit, new TimeUnitRule(unit), UNIT_LENGTHS.get(unit), UNIT_LENGTHS.keySet()); } builder.appendElement( LongElement.POSIX_TIME, LongElement.POSIX_TIME, TimeUnit.SECONDS); builder.appendElement( IntElement.FRACTION, IntElement.FRACTION, TimeUnit.NANOSECONDS); builder.appendElement( PrecisionElement.TIME_PRECISION, new PrecisionRule()); ENGINE = builder.withTimeLine(new GlobalTimeLine()).build(); } /** *

Start of UNIX-era = [1970-01-01T00:00:00,000000000Z].

*/ /*[deutsch] *

Start der UNIX-Ära = [1970-01-01T00:00:00,000000000Z].

*/ public static final Moment UNIX_EPOCH = new Moment(0, 0, TimeScale.POSIX); /** *

Represents the POSIX-time in seconds since UNIX-epoch.

* * @since 2.0 */ /*[deutsch] *

Repräsentiert die POSIX-Zeit in Sekunden seit der * UNIX-Epoche.

* * @since 2.0 */ public static final ChronoElement POSIX_TIME = LongElement.POSIX_TIME; /** *

Represents the nano-fraction of current second.

* * @since 2.0 */ /*[deutsch] *

Repräsentiert den Nanosekundenbruchteil der aktuellen * Sekunde.

* * @since 2.0 */ public static final ChronoElement FRACTION = IntElement.FRACTION; /** *

Represents the precision.

* * @since 3.7/4.5 */ /*[deutsch] *

Repräsentiert die Genauigkeit.

* * @since 3.7/4.5 */ public static final ChronoElement PRECISION = PrecisionElement.TIME_PRECISION; private static final ChronoOperator NEXT_LS = new NextLS(); private static final long serialVersionUID = -3192884724477742274L; //~ Instanzvariablen -------------------------------------------------- private transient final long posixTime; private transient final int fraction; //~ Konstruktoren ----------------------------------------------------- private Moment( long elapsedTime, int nanosecond, TimeScale scale ) { super(); if (scale == POSIX) { this.posixTime = elapsedTime; this.fraction = nanosecond; } else { LeapSeconds ls = LeapSeconds.getInstance(); if (ls.isEnabled()) { long utcTime; if (scale == UTC) { utcTime = elapsedTime; } else if (scale == TAI) { utcTime = Math.subtractExact(elapsedTime, 10); if (utcTime < 0) { throw new IllegalArgumentException( "TAI not supported before 1972-01-01: " + elapsedTime); } } else if (scale == GPS) { utcTime = Math.addExact(elapsedTime, UTC_GPS_DELTA); if (utcTime < UTC_GPS_DELTA) { throw new IllegalArgumentException( "GPS not supported before 1980-01-06: " + elapsedTime); } } else { throw new UnsupportedOperationException( "Not yet implemented: " + scale.name()); } long unix = ls.strip(utcTime); long diff = (utcTime - ls.enhance(unix)); this.posixTime = unix; if ( (diff == 0) || (unix == MAX_LIMIT) ) { this.fraction = nanosecond; } else if (diff == 1) { // positive Schaltsekunde this.fraction = (nanosecond | POSITIVE_LEAP_MASK); } else { throw new IllegalStateException( "Cannot handle leap shift of " + elapsedTime + "."); } } else { throw new IllegalStateException( "Leap seconds are not supported by configuration."); } } checkUnixTime(this.posixTime); checkFraction(nanosecond); } // Deserialisierung private Moment( int nano, long unixTime ) { super(); // keine Prüfung des Nano-Anteils und Schaltsekunden-Bits checkUnixTime(unixTime); this.posixTime = unixTime; this.fraction = nano; } //~ Methoden ---------------------------------------------------------- /** *

Equivalent to {@code Moment.of(elapsedTime, 0, scale)}.

* * @param elapsedTime elapsed seconds on given time scale * @param scale time scale reference * @return new moment instance * @throws IllegalArgumentException if elapsed time is out of range limits * beyond year +/-999,999,999 or out of time scale range * @throws IllegalStateException if time scale is not POSIX but * leap second support is switched off by configuration * @see LeapSeconds#isEnabled() */ /*[deutsch] *

Entspricht {@code Moment.of(elapsedTime, 0, scale)}.

* * @param elapsedTime elapsed seconds on given time scale * @param scale time scale reference * @return new moment instance * @throws IllegalArgumentException if elapsed time is out of range limits * beyond year +/-999,999,999 or out of time scale range * @throws IllegalStateException if time scale is not POSIX but * leap second support is switched off by configuration * @see LeapSeconds#isEnabled() */ public static Moment of( long elapsedTime, TimeScale scale ) { return Moment.of(elapsedTime, 0, scale); } /** *

Creates a new UTC-timestamp by given time coordinates on given * time scale.

* *

The given elapsed time {@code elapsedTime} will be internally * transformed into the UTC-epochtime, should another time scale than UTC * be given. The time scale TAI will only be supported earliest on UTC * start 1972-01-01, the time scale GPS earliest on 1980-01-06.

* * @param elapsedTime elapsed seconds on given time scale * @param nanosecond nanosecond fraction of last second * @param scale time scale reference * @return new moment instance * @throws IllegalArgumentException if the nanosecond is not in the range * {@code 0 <= nanosecond <= 999,999,999} or if elapsed time is * out of supported range limits beyond year +/-999,999,999 or * out of time scale range * @throws IllegalStateException if time scale is not POSIX but * leap second support is switched off by configuration * @see LeapSeconds#isEnabled() */ /*[deutsch] *

Konstruiert einen neuen UTC-Zeitstempel mit Hilfe von * Zeitkoordinaten auf der angegebenen Zeitskala.

* *

Die angegebene verstrichene Zeit {@code elapsedTime} wird intern * in die UTC-Epochenzeit umgerechnet, sollte eine andere Zeitskala als * UTC angegeben sein. Die Zeitskala TAI wird erst ab der UTC-Epoche * 1972-01-01 unterstützt, die Zeitskala GPS erst ab 1980-01-06.

* * @param elapsedTime elapsed seconds on given time scale * @param nanosecond nanosecond fraction of last second * @param scale time scale reference * @return new moment instance * @throws IllegalArgumentException if the nanosecond is not in the range * {@code 0 <= nanosecond <= 999,999,999} or if elapsed time is * out of supported range limits beyond year +/-999,999,999 or * out of time scale range * @throws IllegalStateException if time scale is not POSIX but * leap second support is switched off by configuration * @see LeapSeconds#isEnabled() */ public static Moment of( long elapsedTime, int nanosecond, TimeScale scale ) { if ( (elapsedTime == 0) && (nanosecond == 0) && (scale == POSIX) ) { return Moment.UNIX_EPOCH; } return new Moment(elapsedTime, nanosecond, scale); } /** *

Obtains the current time using the system clock.

* *

Equivalent alternative for: {@code SystemClock.currentMoment()}.

* * @return current moment using the system clock * @see SystemClock#currentMoment() * @since 3.23/4.19 */ /*[deutsch] *

Ermittelt die aktuelle Systemzeit.

* *

Alternative für: {@code SystemClock.currentMoment()}.

* * @return current moment using the system clock * @see SystemClock#currentMoment() * @since 3.23/4.19 */ public static Moment nowInSystemTime() { return SystemClock.INSTANCE.currentTime(); } /** *

Common conversion method.

* * @param ut UNIX-timestamp * @return corresponding {@code Moment} */ /*[deutsch] *

Allgemeine Konversionsmethode.

* * @param ut UNIX-timestamp * @return corresponding {@code Moment} */ public static Moment from(UnixTime ut) { if (ut instanceof Moment) { return Moment.class.cast(ut); } else if ( (ut instanceof UniversalTime) && LeapSeconds.getInstance().isEnabled() ) { UniversalTime utc = UniversalTime.class.cast(ut); return Moment.of( utc.getElapsedTime(UTC), utc.getNanosecond(UTC), UTC); } else { return Moment.of( ut.getPosixTime(), ut.getNanosecond(), POSIX); } } /** *

Short cut for {@code TemporalType.INSTANT.translate(instant)}.

* * @param instant Threeten-equivalent of this instance * @return Moment * @since 4.0 * @see TemporalType#INSTANT */ /*[deutsch] *

Abkürzung für {@code TemporalType.INSTANT.translate(instant)}.

* * @param instant Threeten-equivalent of this instance * @return Moment * @since 4.0 * @see TemporalType#INSTANT */ public static Moment from(Instant instant) { return TemporalType.INSTANT.translate(instant); } @Override public long getPosixTime() { return this.posixTime; } @Override public long getElapsedTime(TimeScale scale) { if (scale == POSIX) { return this.posixTime; } long utc = this.getEpochTime(); switch (scale) { case UTC: return utc; case TAI: if (utc < 0) { throw new IllegalArgumentException( "TAI not supported before 1972-01-01: " + this); } else { return utc + 10; } case GPS: if (LeapSeconds.getInstance().strip(utc) < POSIX_GPS_DELTA) { throw new IllegalArgumentException( "GPS not supported before 1980-01-06: " + this); } else { long gps = LeapSeconds.getInstance().isEnabled() ? utc : (utc + 9); return gps - UTC_GPS_DELTA; } default: throw new UnsupportedOperationException( "Not yet implemented: " + scale); } } @Override public int getNanosecond() { return (this.fraction & (~POSITIVE_LEAP_MASK)); } @Override public int getNanosecond(TimeScale scale) { switch (scale) { case POSIX: case UTC: return this.getNanosecond(); case TAI: if (this.posixTime < POSIX_UTC_DELTA) { throw new IllegalArgumentException( "TAI not supported before 1972-01-01: " + this); } else { return this.getNanosecond(); } case GPS: long utc = this.getEpochTime(); if (LeapSeconds.getInstance().strip(utc) < POSIX_GPS_DELTA) { throw new IllegalArgumentException( "GPS not supported before 1980-01-06: " + this); } else { return this.getNanosecond(); } default: throw new UnsupportedOperationException( "Not yet implemented: " + scale); } } @Override public boolean isLeapSecond() { return (this.isPositiveLS() && LeapSeconds.getInstance().isEnabled()); } /** *

Tries to determine the next coming leap second.

* * @return operator which either gets next leap second or {@code null} * if unknown or disabled * @since 2.1 */ /*[deutsch] *

Versucht, die nächste bevorstehende UTC-Schaltsekunde zu * ermitteln.

* * @return operator which either gets next leap second or {@code null} * if unknown or disabled * @since 2.1 */ public static ChronoOperator nextLeapSecond() { return NEXT_LS; } /** *

Represents this timestamp as decimal value in given time scale.

* *

The scale determines the epoch reference to be used and how to count elapsed seconds * since a given epoch. Please note that some scales like TAI, GPS and UTC are atomic scales * counting SI-seconds continuously. Such time scales do not suppress leap seconds in this view. * However, the scales handle the representation of leap seconds in a very different way. * TAI for example does not consider leap seconds as leap seconds but just normal seconds * (in the model of 1 TAI-day = 86400 SI-seconds) while UTC lables leap seconds with the * special value 60 causing discontinuities.

* * @param scale time scale reference * @return decimal value in given time scale as seconds inclusive fraction * @throws IllegalArgumentException if this instance is out of range for given time scale * @see #toString(TimeScale) */ /*[deutsch] *

Stellt diese Zeit als Dezimalwert in der angegebenen Zeitskala * dar.

* *

Die Zeitskala bestimmt die Epochenreferenz und auch, wie seit dieser Epoche verstrichene * Sekunden gezählt werden. Zu beachten: Einige Zeitskalen wie TAI, GPS und UTC sind atomare * Zeitskalen, die auf Atomuhren Bezug nehmen und kontinuierlich SI-Sekunden zählen. Aber: * Die jeweiligen Zeitskalen behandeln die Repräsentation von Schaltsekunden sehr verschieden. * TAI zum Beispiel betrachtet eine verstrichene Schaltsekunde nicht als Schaltsekunde, sondern * einfach als erste Sekunde des nächsten TAI-Tags, der entsprechend immer 86400 SI-Sekunden * lang ist. Im Kontrast hierzu etikettiert UTC Schaltsekunden als solche, indem sie mit dem * speziellen Wert "60" versehen werden, was eine Diskontinuität darstellt.

* * @param scale time scale reference * @return decimal value in given time scale as seconds inclusive fraction * @throws IllegalArgumentException if this instance is out of range for given time scale * @see #toString(TimeScale) */ public BigDecimal transform(TimeScale scale) { BigDecimal elapsedTime = new BigDecimal(this.getElapsedTime(scale)).setScale(9, RoundingMode.UNNECESSARY); BigDecimal nanosecond = new BigDecimal(this.getNanosecond(scale)); return elapsedTime.add(nanosecond.movePointLeft(9)); } @Override public boolean isAfter(UniversalTime temporal) { Moment other = Moment.from(temporal); return (this.compareTo(other) > 0); } @Override public boolean isBefore(UniversalTime temporal) { Moment other = Moment.from(temporal); return (this.compareTo(other) < 0); } @Override public boolean isSimultaneous(UniversalTime temporal) { Moment other = Moment.from(temporal); return (this.compareTo(other) == 0); } /** *

Converts this instance to a local timestamp in the system * timezone.

* * @return local timestamp in system timezone (leap seconds will * always be lost) * @since 1.2 * @see Timezone#ofSystem() * @see #toZonalTimestamp(TZID) * @see #toZonalTimestamp(String) */ /*[deutsch] *

Wandelt diese Instanz in einen lokalen Zeitstempel um.

* * @return local timestamp in system timezone (leap seconds will * always be lost) * @since 1.2 * @see Timezone#ofSystem() * @see #toZonalTimestamp(TZID) * @see #toZonalTimestamp(String) */ public PlainTimestamp toLocalTimestamp() { return this.in(Timezone.ofSystem()); } /** *

Converts this instance to a local timestamp in given timezone.

* * @param tzid timezone id * @return local timestamp in given timezone (leap seconds will * always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 1.2 * @see #toLocalTimestamp() */ /*[deutsch] *

Wandelt diese Instanz in einen lokalen Zeitstempel um.

* * @param tzid timezone id * @return local timestamp in given timezone (leap seconds will * always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 1.2 * @see #toLocalTimestamp() */ public PlainTimestamp toZonalTimestamp(TZID tzid) { return this.in(Timezone.of(tzid)); } /** *

Converts this instance to a local timestamp in given timezone.

* * @param tzid timezone id * @return local timestamp in given timezone (leap seconds will * always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 1.2 * @see #toZonalTimestamp(TZID) * @see #toLocalTimestamp() */ /*[deutsch] *

Wandelt diese Instanz in einen lokalen Zeitstempel um.

* * @param tzid timezone id * @return local timestamp in given timezone (leap seconds will * always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 1.2 * @see #toZonalTimestamp(TZID) * @see #toLocalTimestamp() */ public PlainTimestamp toZonalTimestamp(String tzid) { return this.in(Timezone.of(tzid)); } /** *

Converts this instance to a general timestamp in given timezone.

* * @param generic type of date component * @param chronology chronology of date component * @param tzid timezone id * @param startOfDay start of day * @return general timestamp in given timezone (leap seconds will always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 3.8/4.5 */ /*[deutsch] *

Wandelt diese Instanz in einen allgemeinen Zeitstempel um.

* * @param generic type of date component * @param chronology chronology of date component * @param tzid timezone id * @param startOfDay start of day * @return general timestamp in given timezone (leap seconds will always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @since 3.8/4.5 */ public > GeneralTimestamp toGeneralTimestamp( Chronology chronology, TZID tzid, StartOfDay startOfDay ) { PlainTimestamp tsp = this.toZonalTimestamp(tzid); PlainTime time = tsp.getWallTime(); int deviation = startOfDay.getDeviation(tsp.getCalendarDate(), tzid); tsp = tsp.minus(deviation, ClockUnit.SECONDS); C date = tsp.getCalendarDate().transform(chronology.getChronoType()); return GeneralTimestamp.of(date, time); } /** *

Converts this instance to a general timestamp in given timezone.

* * @param generic type of date component * @param family calendar family for date component * @param variant variant of date component * @param tzid timezone id * @param startOfDay start of day * @return general timestamp in given timezone (leap seconds will always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @throws ChronoException if given variant is not recognized * @since 3.8/4.5 */ /*[deutsch] *

Wandelt diese Instanz in einen allgemeinen Zeitstempel um.

* * @param generic type of date component * @param family calendar family for date component * @param variant variant of date component * @param tzid timezone id * @param startOfDay start of day * @return general timestamp in given timezone (leap seconds will always be lost) * @throws IllegalArgumentException if given timezone cannot be loaded * @throws ChronoException if given variant is not recognized * @since 3.8/4.5 */ public > GeneralTimestamp toGeneralTimestamp( CalendarFamily family, String variant, TZID tzid, StartOfDay startOfDay ) { PlainTimestamp tsp = this.toZonalTimestamp(tzid); PlainTime time = tsp.getWallTime(); int deviation = startOfDay.getDeviation(tsp.getCalendarDate(), tzid); tsp = tsp.minus(deviation, ClockUnit.SECONDS); C date = tsp.getCalendarDate().transform(family.getChronoType(), variant); return GeneralTimestamp.of(date, time); } /** *

Creates a combination of this moment and system timezone.

* *

A direct conversion to a local timestamp can be achieved by * {@link #toLocalTimestamp()}.

* * @return moment in system timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset */ /*[deutsch] *

Erzeugt eine Kombination dieses Moments und der Systemzeitzone.

* *

Eine Direktumwandlung zu einem lokalen Zeitstempel kann mit Hilfe * von {@link #toLocalTimestamp()} erreicht werden.

* * @return moment in system timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset */ public ZonalDateTime inLocalView() { return ZonalDateTime.of(this, Timezone.ofSystem()); } /** *

Creates a combination of this moment and given timezone.

* *

A direct conversion to a zonal timestamp can be achieved by * {@link #toZonalTimestamp(TZID)}.

* * @param tzid timezone id * @return moment in given timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset or * if given timezone cannot be loaded */ /*[deutsch] *

Erzeugt eine Kombination dieses Moments und der angegebenen * Zeitzone.

* *

Eine Direktumwandlung zu einem zonalen Zeitstempel kann mit Hilfe * von {@link #toZonalTimestamp(TZID)} erreicht werden.

* * @param tzid timezone id * @return moment in given timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset or * if given timezone cannot be loaded */ public ZonalDateTime inZonalView(TZID tzid) { return ZonalDateTime.of(this, Timezone.of(tzid)); } /** *

Creates a combination of this moment and given timezone.

* *

A direct conversion to a zonal timestamp can be achieved by * {@link #toZonalTimestamp(String)}.

* * @param tzid timezone id * @return moment in given timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset or * if given timezone cannot be loaded */ /*[deutsch] *

Erzeugt eine Kombination dieses Moments und der angegebenen * Zeitzone.

* *

Eine Direktumwandlung zu einem zonalen Zeitstempel kann mit Hilfe * von {@link #toZonalTimestamp(String)} erreicht werden.

* * @param tzid timezone id * @return moment in given timezone * @since 2.0 * @throws IllegalArgumentException if this moment is a leapsecond and * shall be combined with a non-full-minute-timezone-offset or * if given timezone cannot be loaded */ public ZonalDateTime inZonalView(String tzid) { return ZonalDateTime.of(this, Timezone.of(tzid)); } /** *

Adds an amount of given SI-unit to this timestamp * on the UTC time scale.

* * @param amount amount in units to be added * @param unit time unit defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow */ /*[deutsch] *

Addiert einen Betrag in der angegegebenen SI-Zeiteinheit auf die * UTC-Zeit dieses Zeitstempels.

* * @param amount amount in units to be added * @param unit time unit defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow */ public Moment plus( long amount, SI unit ) { Moment.check1972(this); if (amount == 0) { return this; } Moment result; try { switch (unit) { case SECONDS: if (LeapSeconds.getInstance().isEnabled()) { result = new Moment( Math.addExact(this.getEpochTime(), amount), this.getNanosecond(), UTC); } else { result = Moment.of( Math.addExact(this.posixTime, amount), this.getNanosecond(), POSIX ); } break; case NANOSECONDS: long sum = Math.addExact(this.getNanosecond(), amount); int nano = (int) Math.floorMod(sum, MRD); long second = Math.floorDiv(sum, MRD); if (LeapSeconds.getInstance().isEnabled()) { result = new Moment( Math.addExact(this.getEpochTime(), second), nano, UTC ); } else { result = Moment.of( Math.addExact(this.posixTime, second), nano, POSIX ); } break; default: throw new UnsupportedOperationException(); } } catch (IllegalArgumentException iae) { ArithmeticException ex = new ArithmeticException( "Result beyond boundaries of time axis."); ex.initCause(iae); throw ex; } if (amount < 0) { Moment.check1972(result); } return result; } /** *

Adds given real time to this timestamp on the UTC time scale.

* * @param realTime real time defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow * @since 3.23/4.19 */ /*[deutsch] *

Addiert die angegebene Realzeit zur UTC-Zeit dieses Zeitstempels.

* * @param realTime real time defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow * @since 3.23/4.19 */ public Moment plus(RealTime realTime) { return this.plus(realTime.getSeconds(), SI.SECONDS).plus(realTime.getFraction(), SI.NANOSECONDS); } /** *

Subtracts an amount of given SI-unit from this timestamp * on the UTC time scale.

* * @param amount amount in SI-units to be subtracted * @param unit time unit defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow */ /*[deutsch] *

Subtrahiert einen Betrag in der angegegebenen Zeiteinheit von der * UTC-Zeit dieses Zeitstempels.

* * @param amount amount in SI-units to be subtracted * @param unit time unit defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow */ public Moment minus( long amount, SI unit ) { return this.plus(Math.negateExact(amount), unit); } /** *

Subtracts given real time from this timestamp on the UTC time scale.

* * @param realTime real time defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow * @since 3.23/4.19 */ /*[deutsch] *

Subtrahiert die angegegebene Realzeit von der UTC-Zeit dieses Zeitstempels.

* * @param realTime real time defined in UTC time space * @return changed copy of this instance * @throws UnsupportedOperationException if either this moment or the result are before 1972 * @throws ArithmeticException in case of overflow * @since 3.23/4.19 */ public Moment minus(RealTime realTime) { return this.minus(realTime.getSeconds(), SI.SECONDS).minus(realTime.getFraction(), SI.NANOSECONDS); } /** *

Calculates the time distance between this timestamp and given * end timestamp in given SI-unit on the UTC time scale.

* * @param end end time point * @param unit time unit defined in UTC time space * @return count of SI-units between this instance and end time point * @throws UnsupportedOperationException if any moment is before 1972 */ /*[deutsch] *

Bestimmt den zeitlichen Abstand zu einem Endzeitpunkt in der * angegebenen Zeiteinheit auf der UTC-Zeitskala.

* * @param end end time point * @param unit time unit defined in UTC time space * @return count of SI-units between this instance and end time point * @throws UnsupportedOperationException wenn ein Zeitpunkt vor 1972 ist */ public long until( Moment end, SI unit ) { return unit.between(this, end); } /** *

Creates a new formatter which uses the given pattern in the * default locale for formatting and parsing UTC-timestamps.

* *

Note: The formatter can be adjusted to other locales and timezones however.

* * @param

generic pattern type * @param formatPattern format definition as pattern * @param patternType pattern dialect * @return format object for formatting {@code Moment}-objects using system locale and system timezone * @throws IllegalArgumentException if resolving of pattern fails * @since 3.0 */ /*[deutsch] *

Erzeugt ein neues Format-Objekt mit Hilfe des angegebenen Musters * in der Standard-Sprach- und Ländereinstellung und in der * System-Zeitzone.

* *

Das Format-Objekt kann an andere Sprachen oder Zeitzonen angepasst werden.

* * @param

generic pattern type * @param formatPattern format definition as pattern * @param patternType pattern dialect * @return format object for formatting {@code Moment}-objects using system locale and system timezone * @throws IllegalArgumentException if resolving of pattern fails * @since 3.0 */ public static

> TemporalFormatter localFormatter( String formatPattern, P patternType ) { return FormatSupport.createFormatter( Moment.class, formatPattern, patternType, Locale.getDefault(), Timezone.ofSystem().getID()); } /** *

Creates a new formatter which uses the given display mode in the * default locale for formatting and parsing UTC-timestamps.

* *

Note: The formatter can be adjusted to other locales and timezones however.

* * @param mode common formatting style for date part and time part * @return format object for formatting {@code Moment}-objects using system locale and system timezone * @throws IllegalStateException if format pattern cannot be retrieved * @since 3.0 */ /*[deutsch] *

Erzeugt ein neues Format-Objekt mit Hilfe des angegebenen Stils * in der Standard-Sprach- und Ländereinstellung und in der * System-Zeitzone.

* *

Das Format-Objekt kann an andere Sprachen oder Zeitzonen angepasst werden.

* * @param mode common formatting style for date part and time part * @return format object for formatting {@code Moment}-objects using system locale and system timezone * @throws IllegalStateException if format pattern cannot be retrieved * @since 3.0 */ public static TemporalFormatter localFormatter(DisplayMode mode) { return formatter(mode, Locale.getDefault(), Timezone.ofSystem().getID()); } /** *

Creates a new formatter which uses the given pattern and locale * for formatting and parsing moments in given timezone.

* *

Note: The given timezone will be used in printing while it serves * as replacement value during parsing (if the text is missing a timezone * information). The formatter can be adjusted to other locales and * timezones however.

* * @param

generic pattern type * @param formatPattern format definition as pattern * @param patternType pattern dialect * @param locale locale setting * @param tzid timezone id * @return format object for formatting {@code Moment}-objects using given locale and timezone * @throws IllegalArgumentException if resolving of pattern fails * @since 3.0 * @see #localFormatter(String,ChronoPattern) */ /*[deutsch] *

Erzeugt ein neues Format-Objekt mit Hilfe des angegebenen Musters in * der angegebenen Sprach- und Ländereinstellung sowie Zeitzone.

* *

Hinweis: Das Format-Objekt kann an andere Sprachen oder Zeitzonen * angepasst werden. Die angegebene Zeitzone dient beim Parsen als * Ersatzwert (wenn im zu interpretierenden Text keine Zeitzoneninformation * existiert).

* * @param

generic pattern type * @param formatPattern format definition as pattern * @param patternType pattern dialect * @param locale locale setting * @param tzid timezone id * @return format object for formatting {@code Moment}-objects using given locale and timezone * @throws IllegalArgumentException if resolving of pattern fails * @since 3.0 * @see #localFormatter(String,ChronoPattern) */ public static

> TemporalFormatter formatter( String formatPattern, P patternType, Locale locale, TZID tzid ) { return FormatSupport.createFormatter(Moment.class, formatPattern, patternType, locale, tzid); } /** *

Creates a new formatter which uses the given display mode and locale * for formatting and parsing moments in given timezone.

* *

Note: The given timezone will be used in printing while it serves * as replacement value during parsing (if the text is missing a timezone * information). The formatter can be adjusted to other locales and * timezones however.

* * @param mode common formatting style for date part and time part * @param locale locale setting * @param tzid timezone id * @return format object for formatting {@code Moment}-objects using given locale * @throws IllegalStateException if format pattern cannot be retrieved * @since 3.0 * @see #localFormatter(DisplayMode) */ /*[deutsch] *

Erzeugt ein neues Format-Objekt mit Hilfe des angegebenen Stils * und in der angegebenen Sprach- und Ländereinstellung und * Zeitzone.

* *

Hinweis: Das Format-Objekt kann an andere Sprachen oder Zeitzonen * angepasst werden. Die angegebene Zeitzone dient beim Parsen als * Ersatzwert (wenn im zu interpretierenden Text keine Zeitzoneninformation * existiert).

* * @param mode common formatting style for date part and time part * @param locale locale setting * @param tzid timezone id * @return format object for formatting {@code Moment}-objects using given locale * @throws IllegalStateException if format pattern cannot be retrieved * @since 3.0 * @see #localFormatter(DisplayMode) */ public static TemporalFormatter formatter( DisplayMode mode, Locale locale, TZID tzid ) { String formatPattern = CalendarText.patternForMoment(mode, mode, locale); return FormatSupport.createFormatter(Moment.class, formatPattern, locale, tzid); } /** *

Defines the RFC-1123-format which is for example used in mail * headers.

* *

Equivalent to the pattern "[EEE, ]d MMM yyyy HH:mm[:ss] XX" * where the timezone offset XX is modified such that in case of zero * offset the expression "GMT" is preferred. "UT" or "Z" * will be accepted as zero offset, too. The text elements will always be interpreted * in English and are case-insensitive. If no extra timezone is specified then this * formatter will use the timezone UTC as default for printing.

* *

Note: In contrast to the RFC-1123-standard this method does not * support military timezone abbreviations (A-Y) or north-american * timezone names (EST, EDT, CST, CDT, MST, MDT, PST, PDT).

* * @return formatter object for RFC-1123 (technical internet-timestamp) */ /*[deutsch] *

Definiert das RFC-1123-Format, das zum Beispiel in Mail-Headers * verwendet wird.

* *

Entspricht "[EEE, ]d MMM yyyy HH:mm[:ss] XX", wobei * der Zeitzonen-Offset XX so modifiziert ist, daß im Fall eines * Null-Offsets bevorzugt der Ausdruck "GMT" benutzt wird. Als * Null-Offset werden auch "UT" oder "Z" akzeptiert. * Die Textelemente werden ohne Beachtung der Groß- oder * Kleinschreibung in Englisch interpretiert. Wird keine extra Zeitzone * angegeben, wird dieser Formatierer die UTC-Zeitzone als Vorgabe zum * Formatieren verwenden.

* *

Zu beachten: Im Gegensatz zum RFC-1123-Standard unterstützt die * Methode keine militärischen Zeitzonen (A-Y) oder nordamerikanischen * Zeitzonennamen (EST, EDT, CST, CDT, MST, MDT, PST, PDT).

* * @return formatter object for RFC-1123 (technical internet-timestamp) */ public static TemporalFormatter formatterRFC1123() { return RFC1123.FORMATTER; // lazy initialization } @Override public int compareTo(Moment moment) { long u1 = this.getEpochTime(); long u2 = moment.getEpochTime(); if (u1 < u2) { return -1; } else if (u1 > u2) { return 1; } else { int result = this.getNanosecond() - moment.getNanosecond(); return ((result > 0) ? 1 : ((result < 0) ? -1 : 0)); } } @Override public boolean equals(Object obj) { if (this == obj) { return true; } else if (obj instanceof Moment) { Moment that = (Moment) obj; if (this.posixTime != that.posixTime) { return false; } if (LeapSeconds.getInstance().isEnabled()) { return (this.fraction == that.fraction); } else { return (this.getNanosecond() == that.getNanosecond()); } } else { return false; } } @Override public int hashCode() { long value = (this.posixTime ^ (this.posixTime >>> 32)); return (19 * ((int) value) + 37 * this.getNanosecond()); } /** *

Provides a canonical representation in the ISO-format * [yyyy-MM-ddTHH:mm:ss,fffffffffZ].

* *

The fraction will only be printed if not zero. Example: * The expression {@code Moment.of(1341100824, 210, TimeScale.UTC)} * has the representation "2012-06-30T23:59:60,000000210Z".

* * @return ISO-8601-formatted string */ /*[deutsch] *

Erzeugt eine kanonische Darstellung im ISO-Format * [yyyy-MM-ddTHH:mm:ss,fffffffffZ].

* *

Der fraktionale Teil wird nur ausgegeben, wenn nicht 0. Beispiel: * Der Ausdruck {@code Moment.of(1341100824, 210, TimeScale.UTC)} * hat die Darstellung "2012-06-30T23:59:60,000000210Z".

* * @return ISO-8601-formatted string */ @Override public String toString() { // Datum berechnen PlainDate date = this.getDateUTC(); // Uhrzeit berechnen int timeOfDay = getTimeOfDay(this); int minutes = timeOfDay / 60; int hour = minutes / 60; int minute = minutes % 60; int second = timeOfDay % 60; // LS-Korrektur (negative LS => 59!!!, positive LS => 60) second += LeapSeconds.getInstance().getShift(this.getEpochTime()); StringBuilder sb = new StringBuilder(50); // Datum formatieren sb.append(date); // Separator sb.append('T'); // Uhrzeit formatieren format(hour, 2, sb); sb.append(':'); format(minute, 2, sb); sb.append(':'); format(second, 2, sb); // Fraktionaler Sekundenteil int nano = this.getNanosecond(); if (nano > 0) { sb.append(','); format(nano, 9, sb); } // UTC-Symbol anhängen sb.append('Z'); return sb.toString(); } /** *

Creates a formatted view of this instance taking in account * given time scale.

* *
     *  Moment moment =
     *      PlainDate.of(2012, Month.JUNE, 30)
     *      .at(PlainTime.of(23, 59, 59, 999999999))
     *      .atUTC()
     *      .plus(1, SI.SECONDS); // move to leap second
     *
     *  System.out.println(moment.toString(TimeScale.POSIX));
     *  // Output: POSIX-2012-06-30T23:59:59,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.UTC));
     *  // Output: UTC-2012-06-30T23:59:60,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.TAI));
     *  // Output: TAI-2012-07-01T00:00:34,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.GPS));
     *  // Output: GPS-2012-07-01T00:00:15,999999999Z
     * 
* * @param scale time scale to be used for formatting * @return formatted string with date-time fields in timezone UTC * @throws IllegalArgumentException if this instance is out of range * for given time scale * @see #getElapsedTime(TimeScale) * @see net.time4j.format.Attributes#TIME_SCALE */ /*[deutsch] *

Erzeugt eine formatierte Sicht dieser Instanz unter * Berücksichtigung der angegebenen Zeitskala.

* *
     *  Moment moment =
     *      PlainDate.of(2012, Month.JUNE, 30)
     *      .at(PlainTime.of(23, 59, 59, 999999999))
     *      .atUTC()
     *      .plus(1, SI.SECONDS); // move to leap second
     *
     *  System.out.println(moment.toString(TimeScale.POSIX));
     *  // Ausgabe: POSIX-2012-06-30T23:59:59,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.UTC));
     *  // Ausgabe: UTC-2012-06-30T23:59:60,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.TAI));
     *  // Ausgabe: TAI-2012-07-01T00:00:34,999999999Z
     *
     *  System.out.println(moment.toString(TimeScale.GPS));
     *  // Ausgabe: GPS-2012-07-01T00:00:15,999999999Z
     * 
* * @param scale time scale to be used for formatting * @return formatted string with date-time fields in timezone UTC * @throws IllegalArgumentException if this instance is out of range * for given time scale * @see #getElapsedTime(TimeScale) * @see net.time4j.format.Attributes#TIME_SCALE */ public String toString(TimeScale scale) { StringBuilder sb = new StringBuilder(50); sb.append(scale.name()); sb.append('-'); switch (scale) { case POSIX: sb.append(PlainTimestamp.from(this, ZonalOffset.UTC)); sb.append('Z'); break; case UTC: sb.append(this.toString()); break; case TAI: Moment tai = this.transformForPrint(scale); sb.append(PlainTimestamp.from(tai, ZonalOffset.UTC)); sb.append('Z'); break; case GPS: Moment gps = this.transformForPrint(scale); sb.append(PlainTimestamp.from(gps, ZonalOffset.UTC)); sb.append('Z'); break; default: throw new UnsupportedOperationException(scale.name()); } return sb.toString(); } @Override public Instant toTemporalAccessor() { return TemporalType.INSTANT.from(this); } /** *

Provides a static access to the associated time axis respective * chronology which contains the chronological rules.

* * @return chronological system as time axis (never {@code null}) */ /*[deutsch] *

Liefert die zugehörige Zeitachse, die alle notwendigen * chronologischen Regeln enthält.

* * @return chronological system as time axis (never {@code null}) */ public static TimeAxis axis() { return ENGINE; } /** *

Provides a static access to the associated time axis using the foreign type S.

* * @param foreign temporal type * @param converter type converter * @return chronological system for foreign type * @see TemporalType#INSTANT * @see TemporalType#JAVA_UTIL_DATE * @since 3.24/4.20 */ /*[deutsch] *

Liefert die zugehörige Zeitachse angepasst für den Fremdtyp S.

* * @param foreign temporal type * @param converter type converter * @return chronological system for foreign type * @see TemporalType#INSTANT * @see TemporalType#JAVA_UTIL_DATE * @since 3.24/4.20 */ public static Chronology axis(Converter converter) { return new BridgeChronology<>(converter, ENGINE); } @Override protected TimeAxis getChronology() { return ENGINE; } @Override protected Moment getContext() { return this; } /** *

Prüft, ob eine negative Schaltsekunde vorliegt.

* * @param posixTime UNIX-time in seconds * @param ts local timestamp used for error message * @throws ChronoException if a negative leap second is touched */ static void checkNegativeLS( long posixTime, PlainTimestamp ts ) { LeapSeconds ls = LeapSeconds.getInstance(); if ( ls.supportsNegativeLS() && (ls.strip(ls.enhance(posixTime)) > posixTime) ) { throw new ChronoException( "Illegal local timestamp due to " + "negative leap second: " + ts); } } /** *

Prüft, ob der Zeitpunkt vor 1972 liegt.

* * @param context Prüfzeitpunkt * @throws UnsupportedOperationException wenn der Zeitpunkt vor 1972 ist */ static void check1972(Moment context) { if (context.posixTime < POSIX_UTC_DELTA) { throw new UnsupportedOperationException( "Cannot calculate SI-duration before 1972-01-01."); } } private long getEpochTime() { if (LeapSeconds.getInstance().isEnabled()) { long time = LeapSeconds.getInstance().enhance(this.posixTime); return (this.isPositiveLS() ? time + 1 : time); } else { return this.posixTime - POSIX_UTC_DELTA; } } // Datum in der UTC-Zeitzone private PlainDate getDateUTC() { return PlainDate.of( Math.floorDiv(this.posixTime, 86400), EpochDays.UNIX); } // Uhrzeit in der UTC-Zeitzone (ohne Schaltsekunde) private PlainTime getTimeUTC() { int timeOfDay = getTimeOfDay(this); int minutes = timeOfDay / 60; int hour = minutes / 60; int minute = minutes % 60; int second = timeOfDay % 60; int nano = this.getNanosecond(); return PlainTime.of(hour, minute, second, nano); } private boolean isPositiveLS() { return ((this.fraction >>> 30) != 0); } private boolean isNegativeLS() { LeapSeconds ls = LeapSeconds.getInstance(); if (ls.supportsNegativeLS()) { long ut = this.posixTime; return (ls.strip(ls.enhance(ut)) > ut); } else { return false; } } private static void checkUnixTime(long unixTime) { if ( (unixTime > MAX_LIMIT) || (unixTime < MIN_LIMIT) ) { throw new IllegalArgumentException( "UNIX time (UT1) out of supported range: " + unixTime); } } private static void checkFraction(int nanoFraction) { if ((nanoFraction >= MRD) || (nanoFraction < 0)) { throw new IllegalArgumentException( "Nanosecond out of range: " + nanoFraction); } } private Moment transformForPrint(TimeScale scale) { switch (scale) { case POSIX: if (this.isLeapSecond()) { return new Moment( this.getNanosecond(), this.posixTime ); } else { return this; } case UTC: return this; case TAI: return new Moment( this.getNanosecond(), Math.addExact( this.getElapsedTime(TAI), POSIX_UTC_DELTA) ); case GPS: return new Moment( this.getNanosecond(), Math.addExact( this.getElapsedTime(GPS), POSIX_GPS_DELTA) ); default: throw new UnsupportedOperationException(scale.name()); } } private Moment transformForParse(TimeScale scale) { if (scale == UTC) { return this; } else if (this.isLeapSecond()) { throw new IllegalArgumentException("Leap seconds do not exist on continuous time scale: " + scale); } switch (scale) { case POSIX: return this; case TAI: return new Moment( Math.subtractExact( this.posixTime, POSIX_UTC_DELTA), this.getNanosecond(), scale ); case GPS: return new Moment( Math.subtractExact( this.posixTime, POSIX_GPS_DELTA), this.getNanosecond(), scale ); default: throw new UnsupportedOperationException(scale.name()); } } private static void format( int value, int max, StringBuilder sb ) { int n = 1; for (int i = 0; i < max - 1; i++) { n *= 10; } while ((value < n) && (n >= 10)) { sb.append('0'); n = n / 10; } sb.append(String.valueOf(value)); } // Anzahl der POSIX-Sekunden des Tages private static int getTimeOfDay(Moment context) { return (int) Math.floorMod(context.posixTime, 86400); } // Schaltsekundenkorrektur private static Moment moveEventuallyToLS(Moment adjusted) { PlainDate date = adjusted.getDateUTC(); PlainTime time = adjusted.getTimeUTC(); if ( (LeapSeconds.getInstance().getShift(date) == 1) && (time.getHour() == 23) && (time.getMinute() == 59) && (time.getSecond() == 59) ) { return adjusted.plus(1, SI.SECONDS); } else { return adjusted; } } private PlainTimestamp in(Timezone tz) { return PlainTimestamp.from(this, tz.getOffset(this)); } private static int getMaxSecondOfMinute(Moment context) { int minutes = getTimeOfDay(context) / 60; int second = 59; if (((minutes / 60) == 23) && ((minutes % 60) == 59)) { PlainDate date = context.getDateUTC(); second += LeapSeconds.getInstance().getShift(date); } return second; } /** * @serialData Uses * a dedicated serialization form as proxy. The format * is bit-compressed. Overall until 13 data bytes are used. * The first byte contains in the four most significant bits * the type-ID {@code 4}. The lowest bit is {@code 1} if this * instance is a positive leap second. The bit (2) will be * set if there is a non-zero nanosecond part. After this * header byte eight bytes follow containing the unix time * (as long) and optional four bytes with the fraction part. * * Schematic algorithm: * *
     *  int header = 4;
     *  header <<= 4;
     *
     *  if (isLeapSecond()) {
     *      header |= 1;
     *  }
     *
     *  int fraction = getNanosecond();
     *
     *  if (fraction > 0) {
     *      header |= 2;
     *  }
     *
     *  out.writeByte(header);
     *  out.writeLong(getPosixTime());
     *
     *  if (fraction > 0) {
     *      out.writeInt(fraction);
     *  }
     * 
* * @return replacement object in serialization graph */ private Object writeReplace() { return new SPX(this, SPX.MOMENT_TYPE); } /** * @serialData Blocks because a serialization proxy is required. * @param in object input stream * @throws InvalidObjectException (always) */ private void readObject(ObjectInputStream in) throws IOException { throw new InvalidObjectException("Serialization proxy required."); } /** * Serialisierungsmethode. * * @param out output stream * @throws IOException */ void writeTimestamp(DataOutput out) throws IOException { int header = SPX.MOMENT_TYPE; header <<= 4; if (this.isPositiveLS()) { header |= 1; } int fp = this.getNanosecond(); if (fp > 0) { header |= 2; } out.writeByte(header); out.writeLong(this.posixTime); if (fp > 0) { out.writeInt(fp); } } /** * Deserialisierungsmethode. * * @param in input stream * @param positiveLS positive leap second indicated? * @return deserialized instance * @throws IOException */ static Moment readTimestamp( DataInput in, boolean positiveLS, boolean hasNanos ) throws IOException { long unixTime = in.readLong(); int nano = (hasNanos ? in.readInt() : 0); if (unixTime == 0) { if (positiveLS) { throw new InvalidObjectException( "UTC epoch is no leap second."); } else if (nano == 0) { return UNIX_EPOCH; } } if ( (unixTime == MIN_LIMIT) && (nano == 0) ) { if (positiveLS) { throw new InvalidObjectException("Minimum is no leap second."); } return MIN; } else if ( (unixTime == MAX_LIMIT) && (nano == MRD - 1) ) { if (positiveLS) { throw new InvalidObjectException("Maximum is no leap second."); } return MAX; } else { checkFraction(nano); } if (positiveLS) { LeapSeconds ls = LeapSeconds.getInstance(); if ( !ls.isEnabled() // keep LS-state when propagating to next vm || ls.isPositiveLS(ls.enhance(unixTime) + 1) ) { nano |= POSITIVE_LEAP_MASK; } else { long packed = GregorianMath.toPackedDate(unixTime); int month = GregorianMath.readMonth(packed); int day = GregorianMath.readDayOfMonth(packed); throw new InvalidObjectException( "Not registered as leap second event: " + GregorianMath.readYear(packed) + "-" + ((month < 10) ? "0" : "") + month + ((day < 10) ? "0" : "") + day + " [Please check leap second configurations " + "either of emitter vm or this target vm]" ); } } return new Moment(nano, unixTime); } //~ Innere Klassen ---------------------------------------------------- /** *

Delegiert Anpassungen von {@code Moment}-Instanzen an einen * {@code ChronoOperator} mit Hilfe einer Zeitzone.

* * @doctags.concurrency {immutable} */ static final class Operator implements ChronoOperator { //~ Instanzvariablen ---------------------------------------------- private final ChronoOperator delegate; private final ChronoElement element; private final int type; private final Timezone tz; //~ Konstruktoren ------------------------------------------------- /** *

Erzeugt einen Operator, der einen {@link Moment} mit * Hilfe der Systemzeitzone anpassen kann.

* * @param delegate delegating operator * @param element element reference * @param type operator type */ Operator( ChronoOperator delegate, ChronoElement element, int type ) { super(); this.delegate = delegate; this.element = element; this.type = type; this.tz = null; } /** *

Erzeugt einen Operator, der einen {@link Moment} mit * Hilfe einer Zeitzonenreferenz anpassen kann.

* * @param delegate delegating operator * @param element element reference * @param type operator type * @param tz timezone */ Operator( ChronoOperator delegate, ChronoElement element, int type, Timezone tz ) { super(); this.delegate = delegate; this.element = element; this.type = type; this.tz = tz; } //~ Methoden ------------------------------------------------------ @Override public Moment apply(Moment moment) { Timezone timezone = ( (this.tz == null) ? Timezone.ofSystem() : this.tz); if ( moment.isLeapSecond() && isNonIsoOffset(timezone, moment) ) { throw new IllegalArgumentException( "Leap second can only be adjusted " + " with timezone-offset in full minutes: " + timezone.getOffset(moment)); } // Spezialfall feingranulare Zeitarithmetik in der UTC-Ära if (moment.isAfter(START_LS_CHECK)) { if ( (this.element == SECOND_OF_MINUTE) && (this.type == ElementOperator.OP_NEW_VALUE) && (this.extractValue() == 60) ) { if (moment.isLeapSecond()) { return moment; } else if (isNonIsoOffset(timezone, moment)) { throw new IllegalArgumentException( "Leap second can only be set " + " with timezone-offset in full minutes: " + timezone.getOffset(moment)); } else if (getMaxSecondOfMinute(moment) == 60) { return moment.plus( Math.subtractExact(60, this.extractOld(moment)), SECONDS); } else { throw new IllegalArgumentException( "Leap second invalid in context: " + moment); } } else if ( LOW_TIME_ELEMENTS.containsKey(this.element) && ((this.type == ElementOperator.OP_DECREMENT) || (this.type == ElementOperator.OP_INCREMENT) || (this.type == ElementOperator.OP_LENIENT)) ) { int step = LOW_TIME_ELEMENTS.get(this.element).intValue(); long amount = 1; if (this.type == ElementOperator.OP_DECREMENT) { amount = -1; } else if (this.type == ElementOperator.OP_LENIENT) { long oldValue = this.extractOld(moment); long newValue = this.extractValue(); amount = Math.subtractExact(newValue, oldValue); } switch (step) { case 1: return moment.plus(amount, SECONDS); case 1000: return moment.plus( Math.multiplyExact(MIO, amount), NANOSECONDS); case MIO: return moment.plus( Math.multiplyExact(1000, amount), NANOSECONDS); case MRD: return moment.plus(amount, NANOSECONDS); default: throw new AssertionError(); } } } // lokale Transformation PlainTimestamp ts = moment.in(timezone).with(this.delegate); Moment result = ts.in(timezone); // hier kann niemals die Schaltsekunde erreicht werden if (this.type == ElementOperator.OP_FLOOR) { return result; } // Schaltsekundenprüfung, weil lokale Transformation keine LS kennt if (result.isNegativeLS()) { if (this.tz.getStrategy() == Timezone.STRICT_MODE) { throw new ChronoException( "Illegal local timestamp due to " + "negative leap second: " + ts); } else { return result; } } if ( this.element.isDateElement() || HIGH_TIME_ELEMENTS.contains(this.element) ) { if ( moment.isLeapSecond() || (this.type == ElementOperator.OP_CEILING) ) { return moveEventuallyToLS(result); } } else if (this.element == SECOND_OF_MINUTE) { if ( (this.type == ElementOperator.OP_MAXIMIZE) || (this.type == ElementOperator.OP_CEILING) ) { return moveEventuallyToLS(result); } } else if ( (this.element == MILLI_OF_SECOND) || (this.element == MICRO_OF_SECOND) || (this.element == NANO_OF_SECOND) ) { switch (this.type) { case ElementOperator.OP_NEW_VALUE: case ElementOperator.OP_MINIMIZE: case ElementOperator.OP_MAXIMIZE: case ElementOperator.OP_CEILING: if (moment.isLeapSecond()) { result = result.plus(1, SI.SECONDS); } break; default: // no-op } } return result; } private long extractOld(Moment context) { return Number.class.cast(context.getTimeUTC().get(this.element)).longValue(); } private long extractValue() { Object obj = ValueOperator.class.cast(this.delegate).getValue(); if (obj == null) { throw new IllegalArgumentException("Missing new element value."); } return Number.class.cast(obj).longValue(); } private static boolean isNonIsoOffset( Timezone timezone, Moment context ) { ZonalOffset offset = timezone.getOffset(context); return ( (offset.getFractionalAmount() != 0) || ((offset.getAbsoluteSeconds() % 60) != 0) ); } } private static class TimeUnitRule implements UnitRule { //~ Instanzvariablen ---------------------------------------------- private final TimeUnit unit; //~ Konstruktoren ------------------------------------------------- TimeUnitRule(TimeUnit unit) { super(); this.unit = unit; } //~ Methoden ------------------------------------------------------ @Override public Moment addTo( Moment context, long amount ) { if (this.unit.compareTo(TimeUnit.SECONDS) >= 0) { long secs = Math.multiplyExact(amount, this.unit.toSeconds(1)); return Moment.of( Math.addExact(context.getPosixTime(), secs), context.getNanosecond(), POSIX ); } else { // MILLIS, MICROS, NANOS long nanos = Math.multiplyExact(amount, this.unit.toNanos(1)); long sum = Math.addExact(context.getNanosecond(), nanos); int nano = (int) Math.floorMod(sum, MRD); long second = Math.floorDiv(sum, MRD); return Moment.of( Math.addExact(context.getPosixTime(), second), nano, POSIX ); } } @Override public long between( Moment start, Moment end ) { long delta; if (this.unit.compareTo(TimeUnit.SECONDS) >= 0) { delta = (end.getPosixTime() - start.getPosixTime()); if (delta < 0) { if (end.getNanosecond() > start.getNanosecond()) { delta++; } } else if (delta > 0) { if (end.getNanosecond() < start.getNanosecond()) { delta--; } } } else { // MILLIS, MICROS, NANOS delta = Math.addExact( Math.multiplyExact( Math.subtractExact( end.getPosixTime(), start.getPosixTime() ), MRD ), end.getNanosecond() - start.getNanosecond() ); } switch (this.unit) { case DAYS: delta = delta / 86400; break; case HOURS: delta = delta / 3600; break; case MINUTES: delta = delta / 60; break; case SECONDS: break; case MILLISECONDS: delta = delta / MIO; break; case MICROSECONDS: delta = delta / 1000; break; case NANOSECONDS: break; default: throw new UnsupportedOperationException(this.unit.name()); } return delta; } } private static enum LongElement implements ChronoElement, ElementRule { //~ Statische Felder/Initialisierungen ---------------------------- POSIX_TIME; //~ Methoden ------------------------------------------------------ @Override public Class getType() { return Long.class; } @Override public char getSymbol() { return '\u0000'; } @Override public int compare( ChronoDisplay o1, ChronoDisplay o2 ) { return o1.get(this).compareTo(o2.get(this)); } @Override public Long getDefaultMinimum() { return Long.valueOf(MIN_LIMIT); } @Override public Long getDefaultMaximum() { return Long.valueOf(MAX_LIMIT); } @Override public boolean isDateElement() { return false; } @Override public boolean isTimeElement() { return false; } @Override public boolean isLenient() { return false; } @Override public Long getValue(Moment context) { return Long.valueOf(context.getPosixTime()); } @Override public Long getMinimum(Moment context) { return Long.valueOf(MIN_LIMIT); } @Override public Long getMaximum(Moment context) { return Long.valueOf(MAX_LIMIT); } @Override public boolean isValid( Moment context, Long value ) { if (value == null) { return false; } long val = value.longValue(); return ((val >= MIN_LIMIT) && (val <= MAX_LIMIT)); } @Override public Moment withValue( Moment context, Long value, boolean lenient ) { if (value == null) { throw new IllegalArgumentException("Missing elapsed seconds."); } return Moment.of( value.longValue(), context.getNanosecond(), TimeScale.POSIX); } @Override public ChronoElement getChildAtFloor(Moment context) { return IntElement.FRACTION; } @Override public ChronoElement getChildAtCeiling(Moment context) { return IntElement.FRACTION; } } private static enum IntElement implements ChronoElement, ElementRule { //~ Statische Felder/Initialisierungen ---------------------------- FRACTION; //~ Methoden ------------------------------------------------------ @Override public Class getType() { return Integer.class; } @Override public char getSymbol() { return '\u0000'; } @Override public int compare( ChronoDisplay o1, ChronoDisplay o2 ) { return o1.get(this).compareTo(o2.get(this)); } @Override public Integer getDefaultMinimum() { return Integer.valueOf(0); } @Override public Integer getDefaultMaximum() { return Integer.valueOf(MRD - 1); } @Override public boolean isDateElement() { return false; } @Override public boolean isTimeElement() { return false; } @Override public boolean isLenient() { return false; } @Override public Integer getValue(Moment context) { return Integer.valueOf(context.getNanosecond()); } @Override public Integer getMinimum(Moment context) { return this.getDefaultMinimum(); } @Override public Integer getMaximum(Moment context) { return this.getDefaultMaximum(); } @Override public boolean isValid( Moment context, Integer value ) { if (value == null) { return false; } int val = value.intValue(); return ((val >= 0) && (val < MRD)); } @Override public Moment withValue( Moment context, Integer value, boolean lenient ) { if (value == null) { throw new IllegalArgumentException("Missing fraction value."); } if (LeapSeconds.getInstance().isEnabled()) { return Moment.of( context.getElapsedTime(TimeScale.UTC), value.intValue(), TimeScale.UTC); } else { return Moment.of( context.getPosixTime(), value.intValue(), TimeScale.POSIX); } } @Override public ChronoElement getChildAtFloor(Moment context) { return null; } @Override public ChronoElement getChildAtCeiling(Moment context) { return null; } } private static class Merger implements ChronoMerger { //~ Methoden ------------------------------------------------------ @Override public String getFormatPattern( DisplayStyle style, Locale locale ) { DisplayMode mode = DisplayMode.ofStyle(style.getStyleValue()); return CalendarText.patternForMoment(mode, mode, locale); } @Override public Moment createFrom( TimeSource clock, AttributeQuery attributes ) { return Moment.from(clock.currentTime()); } @Override @Deprecated public Moment createFrom( TemporalAccessor threeten, AttributeQuery attributes ) { if (threeten.isSupported(ChronoField.INSTANT_SECONDS)) { long secs = threeten.getLong(ChronoField.INSTANT_SECONDS); int nano = 0; if (threeten.isSupported(ChronoField.NANO_OF_SECOND)) { nano = threeten.get(ChronoField.NANO_OF_SECOND); } Moment moment = Moment.of(secs, nano, TimeScale.POSIX); if (threeten.query(DateTimeFormatter.parsedLeapSecond())) { moment = moment.plus(1, SI.SECONDS); if (!moment.isLeapSecond()) { throw new IllegalArgumentException("Parsed leap second is invalid."); } } return moment; } else { PlainTimestamp tsp = PlainTimestamp.axis().createFrom(threeten, attributes); if (tsp != null) { TZID tzid = null; if (attributes.contains(Attributes.TIMEZONE_ID)) { tzid = attributes.get(Attributes.TIMEZONE_ID); // Ersatzwert } if (tzid != null) { if (attributes.contains(Attributes.TRANSITION_STRATEGY)) { TransitionStrategy strategy = attributes.get(Attributes.TRANSITION_STRATEGY); return tsp.in(Timezone.of(tzid).with(strategy)); } else { return tsp.inTimezone(tzid); } } } } return null; } @Override @Deprecated public Moment createFrom( ChronoEntity entity, AttributeQuery attributes, boolean preparsing ) { boolean lenient = attributes.get(Attributes.LENIENCY, Leniency.SMART).isLax(); return this.createFrom(entity, attributes, lenient, preparsing); } @Override public Moment createFrom( ChronoEntity entity, AttributeQuery attrs, boolean lenient, boolean pp ) { TimeScale scale = attrs.get(Attributes.TIME_SCALE, TimeScale.UTC); if (entity instanceof UnixTime) { return Moment.from(UnixTime.class.cast(entity)).transformForParse(scale); } else if (entity.contains(LongElement.POSIX_TIME)) { long posixTime = entity.get(LongElement.POSIX_TIME).longValue(); int fraction = 0; if (entity.contains(IntElement.FRACTION)) { fraction = entity.get(IntElement.FRACTION).intValue(); } return Moment.of(posixTime, fraction, POSIX).transformForParse(scale); } Moment result = null; boolean leapsecond = false; if (entity.contains(FlagElement.LEAP_SECOND)) { leapsecond = true; entity.with(SECOND_OF_MINUTE, 60); } ChronoElement self = PlainTimestamp.axis().element(); PlainTimestamp ts; if (entity.contains(self)) { ts = entity.get(self); } else { ts = PlainTimestamp.axis().createFrom(entity, attrs, lenient, pp); } if (ts == null) { return null; } TZID tzid = null; if (entity.hasTimezone()) { tzid = entity.getTimezone(); } else if (attrs.contains(Attributes.TIMEZONE_ID)) { tzid = attrs.get(Attributes.TIMEZONE_ID); // Ersatzwert } if (tzid != null) { if (entity.contains(FlagElement.DAYLIGHT_SAVING)) { boolean dst = entity.get(FlagElement.DAYLIGHT_SAVING).booleanValue(); TransitionStrategy strategy = attrs .get(Attributes.TRANSITION_STRATEGY, Timezone.DEFAULT_CONFLICT_STRATEGY) .using(dst ? OverlapResolver.EARLIER_OFFSET : OverlapResolver.LATER_OFFSET); result = ts.in(Timezone.of(tzid).with(strategy)); } else if (attrs.contains(Attributes.TRANSITION_STRATEGY)) { TransitionStrategy strategy = attrs.get(Attributes.TRANSITION_STRATEGY); result = ts.in(Timezone.of(tzid).with(strategy)); } else { result = ts.inTimezone(tzid); } } if (result == null) { return null; } if (leapsecond) { ZonalOffset offset; if (tzid instanceof ZonalOffset) { offset = (ZonalOffset) tzid; } else { offset = Timezone.of(tzid).getOffset(result); } if ( (offset.getFractionalAmount() != 0) || ((offset.getAbsoluteSeconds() % 60) != 0) ) { throw new IllegalArgumentException( "Leap second is only allowed " + " with timezone-offset in full minutes: " + offset); } Moment test; if (result.getDateUTC().getYear() >= 1972) { test = result.plus(1, SECONDS); } else { test = new Moment( result.getNanosecond(), result.getPosixTime() + 1); } if (lenient) { result = test; } else if (LeapSeconds.getInstance().isEnabled()) { if (test.isPositiveLS()) { result = test; } else { throw new IllegalArgumentException( "SECOND_OF_MINUTE parsed as invalid leapsecond before " + test); } } } return result.transformForParse(scale); } @Override public ChronoDisplay preformat( Moment context, AttributeQuery attributes ) { if (attributes.contains(Attributes.TIMEZONE_ID)) { TZID tzid = attributes.get(Attributes.TIMEZONE_ID); TimeScale scale = attributes.get(Attributes.TIME_SCALE, TimeScale.UTC); return context.transformForPrint(scale).inZonalView(tzid); } throw new IllegalArgumentException("Cannot print moment without timezone."); } @Override public Chronology preparser() { return PlainTimestamp.axis(); } } private static class GlobalTimeLine implements TimeLine { //~ Methoden ------------------------------------------------------ @Override public Moment stepForward(Moment timepoint) { try { if (useSI(timepoint)) { return timepoint.plus(1, SI.NANOSECONDS); } else { return timepoint.plus(1, TimeUnit.NANOSECONDS); } } catch (ArithmeticException iae) { return null; // out of range } } @Override public Moment stepBackwards(Moment timepoint) { try { if (useSI(timepoint)) { return timepoint.minus(1, SI.NANOSECONDS); } else { return timepoint.minus(1, TimeUnit.NANOSECONDS); } } catch (ArithmeticException iae) { return null; // out of range } } @Override public Moment getMinimum() { return MIN; } @Override public Moment getMaximum() { return MAX; } @Override public int compare(Moment m1, Moment m2) { return m1.compareTo(m2); } private static boolean useSI(Moment timepoint) { return ( (timepoint.posixTime > POSIX_UTC_DELTA) && LeapSeconds.getInstance().isEnabled() ); } } private static class NextLS implements ChronoOperator { //~ Methoden ------------------------------------------------------ @Override public Moment apply(Moment timepoint) { LeapSeconds ls = LeapSeconds.getInstance(); if (ls.isEnabled()) { long utc = timepoint.getElapsedTime(TimeScale.UTC); LeapSecondEvent event = ls.getNextEvent(utc); if (event != null) { PlainTimestamp tsp = PlainDate.from(event.getDate()).atTime(23, 59, 59); return tsp.atUTC().plus(event.getShift(), SECONDS); } } return null; } } private static class PrecisionRule implements ElementRule { //~ Methoden ------------------------------------------------------ @Override public TimeUnit getValue(Moment context) { int f = context.getNanosecond(); if (f != 0) { if ((f % MIO) == 0) { return TimeUnit.MILLISECONDS; } else if ((f % 1000) == 0) { return TimeUnit.MICROSECONDS; } else { return TimeUnit.NANOSECONDS; } } long secs = context.posixTime; if (Math.floorMod(secs, 86400) == 0) { return TimeUnit.DAYS; } else if (Math.floorMod(secs, 3600) == 0) { return TimeUnit.HOURS; } else if (Math.floorMod(secs, 60) == 0) { return TimeUnit.MINUTES; } else { return TimeUnit.SECONDS; } } @Override public Moment withValue( Moment context, TimeUnit value, boolean lenient ) { if (value == null) { throw new IllegalArgumentException("Missing precision."); } Moment result; switch (value) { case DAYS: long secsD = Math.floorDiv(context.posixTime, 86400) * 86400; return Moment.of(secsD, TimeScale.POSIX); case HOURS: long secsH = Math.floorDiv(context.posixTime, 3600) * 3600; return Moment.of(secsH, TimeScale.POSIX); case MINUTES: long secsM = Math.floorDiv(context.posixTime, 60) * 60; return Moment.of(secsM, TimeScale.POSIX); case SECONDS: result = Moment.of(context.posixTime, 0, TimeScale.POSIX); break; case MILLISECONDS: int f3 = (context.getNanosecond() / MIO) * MIO; result = Moment.of(context.posixTime, f3, TimeScale.POSIX); break; case MICROSECONDS: int f6 = (context.getNanosecond() / 1000) * 1000; result = Moment.of(context.posixTime, f6, TimeScale.POSIX); break; case NANOSECONDS: return context; default: throw new UnsupportedOperationException(value.name()); } if (context.isLeapSecond() && LeapSeconds.getInstance().isEnabled()) { return result.plus(1, SI.SECONDS); } else { return result; } } @Override public boolean isValid( Moment context, TimeUnit value ) { return (value != null); } @Override public TimeUnit getMinimum(Moment context) { return TimeUnit.DAYS; } @Override public TimeUnit getMaximum(Moment context) { return TimeUnit.NANOSECONDS; } @Override public ChronoElement getChildAtFloor(Moment context) { return null; } @Override public ChronoElement getChildAtCeiling(Moment context) { return null; } } @SuppressWarnings("unchecked") private static class RFC1123 { // Typecast is okay because the type Moment is required per specification. static final TemporalFormatter FORMATTER = (TemporalFormatter) FormatSupport.getDefaultFormatEngine().createRFC1123(); } }