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

net.time4j.tz.Timezone Maven / Gradle / Ivy

There is a newer version: 4.38
Show newest version
/*
 * -----------------------------------------------------------------------
 * Copyright © 2013-2015 Meno Hochschild, 
 * -----------------------------------------------------------------------
 * This file (Timezone.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.tz;

import net.time4j.base.GregorianDate;
import net.time4j.base.ResourceLoader;
import net.time4j.base.UnixTime;
import net.time4j.base.WallTime;

import java.io.IOException;
import java.io.Serializable;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;


/**
 * 

Loads and keeps timezone data including the rules.

* *

Timezones are identified by keys which have canonical forms as * documented in {@link TZID}. If the keys don't specify any provider * (no char "~") then the timezone data and rules will be * looked up using the default {@code ZoneProvider}. This default provider * is loaded by {@code java.util.ServiceLoader} if its name is equal * to "TZDB" and its version string is not empty but of * the highest value (lexicographically). If no such provider can be * found then Time4J uses the platform provider based on the public * API of {@code java.util.TimeZone} which does not expose its * transition history however.

* *

Note: The concept of timezones is strongly based on the idea to * avoid any unrounded amounts of seconds or subseconds. Timezones were * historically first introduced by british railway companies to * guarantee fixed departure timetables. Consequently ISO-8601 only * knows timezone offsets in full minutes. The widely used TZDB-repository * of IANA knows in extreme case offsets in full seconds which is also * allowed by Time4J. Although the Time4J-library recognizes * {@link ZonalOffset#atLongitude(java.math.BigDecimal) fractional offsets} * based on the geographical longitude, its {@code Timezone}-API will * always ignore any fractional parts.

* * @author Meno Hochschild * @serial exclude */ /*[deutsch] *

Lädt und hält Zeitzonendaten mitsamt ihren Regeln.

* *

Zeitzonen werden durch Schlüssel identifiziert, welche eine * kanonische Form wie in {@link TZID} dokumentiert haben. Wenn die * Schlüssel nicht einen spezifischen {@code ZoneProvider} festlegen * (fehlende Tilde "~"), dann werden Zeitzonendaten und Regeln * vom Standard-Provider abgefragt. Dieser wird über einen * {@code java.util.ServiceLoader} geladen, wenn sein Name gleich * "TZDB" ist und seine Version lexikalisch die höchste * und nicht-leer ist. Kann kein solcher {@code ZoneProvider} gefunden * werden, dann verwendet Time4J ersatzweise das öffentliche API von * {@code java.util.TimeZone} (welches allerdings keine Historie * exponiert).

* *

Hinweis: Das Zeitzonenkonzept fußt stark auf der Idee, * irgendwelche nicht-runden Sekunden- oder Subsekundenbeträge * zu vermeiden. Historisch wurden Zeitzonen zuerst von britischen * Eisenbahngesellschaften mit der Motivation eingeführt, landesweit * feste Fahrpläne zu ermöglichen. Konsequenterweise kennt * ISO-8601 nur Zeitzonen-Offsets (Verschiebungen) in vollen Minuten. * Die weit verbreitete Zeitzonendatenbank TZDB von IANA kennt in * Extremfällen auch Offsets in vollen Sekunden, was von Time4J * akzeptiert wird. Obwohl die Time4J-Bibliothek * {@link ZonalOffset#atLongitude(java.math.BigDecimal) fraktionale Offsets} * basierend auf der geographischen Länge kennt, wird das * {@code Timezone}-API immer fraktionale Subsekunden ignorieren.

* * @author Meno Hochschild * @serial exclude */ public abstract class Timezone implements Serializable { //~ Statische Felder/Initialisierungen -------------------------------- private static final String NEW_LINE = System.getProperty("line.separator"); private static final String REPOSITORY_VERSION = System.getProperty("net.time4j.tz.repository.version"); private static final Comparator ID_COMPARATOR = new Comparator() { @Override public int compare( TZID o1, TZID o2 ) { return o1.canonical().compareTo(o2.canonical()); } }; /** *

This standard strategy which is also used by the JDK-class * {@code java.util.GregorianCalendar} subtracts the next defined * offset from any local timestamp in order to calculate the global * time while pushing forward an invalid local time.

* *

Equivalent to * {@code GapResolver.PUSH_FORWARD.and(OverlapResolver.LATER_OFFSET)}.

* * @see #getOffset(GregorianDate,WallTime) */ /*[deutsch] *

Diese auch von der JDK-Klasse {@code java.util.GregorianCalendar} * verwendete Standardstrategie zieht von einem beliebigen lokalen * Zeitstempel den jeweils nächstdefinierten Offset ab, um die * globale Zeit zu erhalten, wobei eine ungültige lokale Zeit * um die Länge einer Offset-Verschiebung vorgeschoben wird.

* *

Äquivalent zu: * {@code GapResolver.PUSH_FORWARD.and(OverlapResolver.LATER_OFFSET)}.

* * @see #getOffset(GregorianDate,WallTime) */ public static final TransitionStrategy DEFAULT_CONFLICT_STRATEGY = GapResolver.PUSH_FORWARD.and(OverlapResolver.LATER_OFFSET); /** *

In addition to the {@link #DEFAULT_CONFLICT_STRATEGY * standard strategy}, this strategy ensures the use of valid local * timestamps.

* *

Equivalent to * {@code GapResolver.ABORT.and(OverlapResolver.LATER_OFFSET)}.

*/ /*[deutsch] *

Legt bei Transformationen von lokalen Zeitstempeln zu UTC fest, * daß nur in der Zeitzone gültige Zeitstempel zugelassen * werden.

* *

Ansonsten wird die {@link #DEFAULT_CONFLICT_STRATEGY * Standardstrategie} verwendet. Äquivalent zu: * {@code GapResolver.ABORT.and(OverlapResolver.LATER_OFFSET)}.

*/ public static final TransitionStrategy STRICT_MODE = GapResolver.ABORT.and(OverlapResolver.LATER_OFFSET); private static final boolean ANDROID = "Dalvik".equalsIgnoreCase(System.getProperty("java.vm.name")); private static final boolean ALLOW_SYSTEM_TZ_OVERRIDE = ANDROID || Boolean.getBoolean("net.time4j.allow.system.tz.override"); private static volatile ZonalKeys zonalKeys = null; private static volatile Timezone currentSystemTZ = null; private static volatile boolean cacheActive = true; private static int softLimit = 11; private static final String NAME_JUT = "java.util.TimeZone"; private static final String NAME_TZDB = "TZDB"; private static final String NAME_ZONENAMES = "#STD_ZONE_NAMES"; private static final String NAME_DEFAULT = "DEFAULT"; private static final Map PREDEFINED; private static final ZoneProvider PLATFORM_PROVIDER; private static final ZoneProvider DEFAULT_PROVIDER; private static final ConcurrentMap CACHE; private static final ReferenceQueue QUEUE; private static final LinkedList LAST_USED; private static final ConcurrentMap PROVIDERS; private static final ZoneProvider ZONENAME_PROVIDER; private static final Timezone SYSTEM_TZ_ORIGINAL; static { CACHE = new ConcurrentHashMap(); PROVIDERS = new ConcurrentHashMap(); QUEUE = new ReferenceQueue(); LAST_USED = new LinkedList(); // strong references List> areas; try { areas = loadPredefined( Timezone.class.getClassLoader(), "AFRICA", "AMERICA", "AMERICA$ARGENTINA", "AMERICA$INDIANA", "AMERICA$KENTUCKY", "AMERICA$NORTH_DAKOTA", "ANTARCTICA", "ASIA", "ATLANTIC", "AUSTRALIA", "EUROPE", "INDIAN", "PACIFIC"); } catch (ClassNotFoundException cnfe) { // olson-package not available areas = Collections.emptyList(); } Map temp1 = new HashMap(); temp1.put("Z", ZonalOffset.UTC); temp1.put("UT", ZonalOffset.UTC); temp1.put("UTC", ZonalOffset.UTC); temp1.put("GMT", ZonalOffset.UTC); temp1.put("UTC0", ZonalOffset.UTC); temp1.put("GMT0", ZonalOffset.UTC); for (Class area : areas) { for (TZID tzid : area.getEnumConstants()) { temp1.put(tzid.canonical(), tzid); } } PREDEFINED = Collections.unmodifiableMap(temp1); ZoneProvider zp = null; ZoneProvider np = null; for (ZoneProvider provider : ResourceLoader.getInstance().services(ZoneProvider.class)) { String name = provider.getName(); if (name.equals(NAME_TZDB)) { zp = compareTZDB(provider, zp); } else if (name.equals(NAME_ZONENAMES)) { np = provider; } else if ( !name.isEmpty() && !name.equals(NAME_DEFAULT) ) { PROVIDERS.put(name, provider); } } ZONENAME_PROVIDER = np; PLATFORM_PROVIDER = new PlatformTZProvider(); PROVIDERS.put(NAME_JUT, PLATFORM_PROVIDER); if (zp == null) { DEFAULT_PROVIDER = PLATFORM_PROVIDER; } else { PROVIDERS.put(NAME_TZDB, zp); DEFAULT_PROVIDER = zp; } Timezone systemTZ = null; try { String zoneID = System.getProperty("user.timezone"); if ( "Z".equals(zoneID) || "UTC".equals(zoneID) ) { systemTZ = ZonalOffset.UTC.getModel(); } else if (zoneID != null) { systemTZ = Timezone.getTZ(resolve(zoneID), zoneID, false); } } catch (SecurityException se) { // OK, dann Zugriff auf j.u.TimeZone.getDefault() } if (systemTZ == null) { SYSTEM_TZ_ORIGINAL = Timezone.getDefaultTZ(); } else { SYSTEM_TZ_ORIGINAL = systemTZ; } if (ALLOW_SYSTEM_TZ_OVERRIDE) { currentSystemTZ = SYSTEM_TZ_ORIGINAL; } // Cache für Available-IDs zonalKeys = new ZonalKeys(); } //~ Konstruktoren ----------------------------------------------------- /** *

Nur zur paket-privaten Verwendung.

*/ Timezone() { super(); } //~ Methoden ---------------------------------------------------------- /** *

Gets all available timezone IDs.

* * @return unmodifiable list of available timezone ids in ascending order */ /*[deutsch] *

Liefert alle verfügbaren Zeitzonenkennungen.

* * @return unmodifiable list of available timezone ids in ascending order */ public static List getAvailableIDs() { return zonalKeys.availables; } /** *

Gets all available timezone IDs for given {@code ZoneProvider}.

* *

Note that this method will return an empty list if given provider * name does not refer to any registered provider. If the name is equal * to "DEFAULT" then the default {@code ZoneProvider} will be * queried.

* * @param provider the registered zone provider whose ids are searched * @return unmodifiable list of available timezone ids in ascending order * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 * @see ZoneProvider#getName() */ /*[deutsch] *

Liefert alle verfügbaren Zeitzonenkennungen zum angegebenen * {@code ZoneProvider}.

* *

Hinweis: Wenn das Argument keinen registrierten {@code ZoneProvider} * referenziert, dann liefert diese Methode eine leere Liste. Wenn das * Argument gleich "DEFAULT" ist, dann wird der * Standard-{@code ZoneProvider} abgefragt.

* * @param provider the registered zone provider whose ids are searched * @return unmodifiable list of available timezone ids in ascending order * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 * @see ZoneProvider#getName() */ public static List getAvailableIDs(String provider) { ZoneProvider zp = getProvider(provider); if (zp == null) { return Collections.emptyList(); } List result = new ArrayList(); for (String id : zp.getAvailableIDs()) { result.add(resolve(id)); } Collections.sort(result, ID_COMPARATOR); return Collections.unmodifiableList(result); } /** *

Gets a provider-specific {@code Set} of preferred timezone IDs * for given ISO-3166-country code.

* *

This information is necessary to enable parsing of timezone names * and is only available if the given provider supports it and denotes * a valid registered provider. Otherwise this method will produce an * empty set. if given provider name is "DEFAULT" then the * default zone provider will be queried.

* * @param locale ISO-3166-alpha-2-country to be evaluated * @param smart if {@code true} then try to select zone ids such * that there is only one preferred id per zone name * @param provider the registered zone provider whose preferred ids * are queried * @return unmodifiable set of preferred timezone ids * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 * @see ZoneProvider#getPreferredIDs(Locale, boolean) */ /*[deutsch] *

Liefert die für einen gegebenen ISO-3166-Ländercode * und {@code ZoneProvider} bevorzugten Zeitzonenkennungen.

* *

Diese Information ist für die Interpretation von Zeitzonennamen * notwendig und steht nur dann zur Verfügung, wenn der angegebene * {@code ZoneProvider} das unterstützt. Wenn hingegen das Argument * {@code provider} keinen registrierten {@code ZoneProvider} referenziert, * liefert diese Methode eine leere Menge. Wenn das Argument gleich * "DEFAULT" ist, dann wird der Standard-{@code ZoneProvider} * abgefragt.

* * @param locale ISO-3166-alpha-2-country to be evaluated * @param smart if {@code true} then try to select zone ids such * that there is only one preferred id per zone name * @param provider the registered zone provider whose preferred ids * are queried * @return unmodifiable set of preferred timezone ids * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 * @see ZoneProvider#getPreferredIDs(Locale, boolean) */ public static Set getPreferredIDs( Locale locale, boolean smart, String provider ) { ZoneProvider zp = getProvider(provider); if (zp == null) { return Collections.emptySet(); } Set p = new HashSet(); for (String id : zp.getPreferredIDs(locale, smart)) { p.add(resolve(id)); } return Collections.unmodifiableSet(p); } /** *

Gets the system timezone.

* *

The underlying algorithm to determine the system timezone is * primarily based on the system property "user.timezone" * then on the method {@code java.util.TimeZone.getDefault()}. If * the system property "net.time4j.allow.system.tz.override" * is set to "true" then the system timezone can be changed * by a combined approach of {@code java.util.TimeZone.setDefault()} * and the method {@link Cache#refresh()}. Otherwise this class * will determine the system timezone only for one time while being * loaded.

* *

Note: If the system timezone cannot be determined (for example * due to a wrong property value for "user.timezone") then * this method will fall back to UTC timezone..

* * @return default timezone data of system * @see java.util.TimeZone#getDefault() * java.util.TimeZone.getDefault() */ /*[deutsch] *

Liefert die Standard-Zeitzone des Systems.

* *

Der verwendete Algorithmus basiert vorrangig auf der * System-Property "user.timezone", dann erst auf der * Methode {@code java.util.TimeZone.getDefault()}. Ist zusätzlich * die Property "net.time4j.allow.system.tz.override" auf den * Wert "true" gesetzt, dann kann nach einer Kombination aus * {@code java.util.TimeZone.setDefault()} und der Methode * {@link Cache#refresh()} in dieser Klasse die Standard-Zeitzone * auch geändert werden, sonst wird sie einmalig beim Laden * dieser Klasse gesetzt.

* *

Zu beachten: Kann die Standard-Zeitzone zum Beispiel wegen eines * falschen Property-Werts in "user.timezone" nicht interpretiert * werden, fällt diese Methode auf die UTC-Zeitzone zurück.

* * @return default timezone data of system * @see java.util.TimeZone#getDefault() * java.util.TimeZone.getDefault() */ public static Timezone ofSystem() { if (ALLOW_SYSTEM_TZ_OVERRIDE && (currentSystemTZ != null)) { return currentSystemTZ; } else { // detect premature class initialization assert (SYSTEM_TZ_ORIGINAL != null); return SYSTEM_TZ_ORIGINAL; } } /** *

Gets the timezone for given identifier.

* *

Queries the underlying {@code ZoneProvider}.

* * @param tzid timezone id as interface * @return timezone data * @throws IllegalArgumentException if given timezone cannot be loaded */ /*[deutsch] *

Liefert die Zeitzone mit der angegebenen ID.

* *

Fragt den zugrundeliegenden {@code ZoneProvider} ab.

* * @param tzid timezone id as interface * @return timezone data * @throws IllegalArgumentException if given timezone cannot be loaded */ public static Timezone of(TZID tzid) { return Timezone.getTZ(tzid, true); } /** *

Gets the timezone for given identifier.

* *

Queries the underlying {@code ZoneProvider}.

* * @param tzid timezone id as String * @return timezone data * @throws IllegalArgumentException if given timezone cannot be loaded */ /*[deutsch] *

Liefert die Zeitzone mit der angegebenen ID.

* *

Fragt den zugrundeliegenden {@code ZoneProvider} ab.

* * @param tzid timezone id as String * @return timezone data * @throws IllegalArgumentException if given timezone cannot be loaded */ public static Timezone of(String tzid) { return Timezone.getTZ(null, tzid, true); } /** *

Tries to load the timezone with the first given identifer else * with given alternative identifier.

* *

If the timezone cannot be loaded with first identifier then * this method will load the timezone using the alternative. In case * of failure, this method will finally load the system timezone. * In contrast to {@link #of(TZID)}, this method never throws any * exception.

* *

Queries the underlying {@code ZoneProvider}.

* * @param tzid preferred timezone id * @param fallback alternative timezone id * @return timezone data */ /*[deutsch] *

Versucht bevorzugt, die Zeitzone mit der angegebenen ID zu laden, * sonst eine Alternative.

* *

Ist die Zeitzone zur ID nicht ladbar, dann wird als zweiter Versuch * die Zeitzone passend zum zweiten Argument geladen. Schlägt auch * das fehl, wird schließlich die Standard-Zeitzone der JVM geladen. * Im Gegensatz zu {@link #of(TZID)} wirft diese Methode niemals eine * Ausnahme.

* *

Fragt den zugrundeliegenden {@code ZoneProvider} ab.

* * @param tzid preferred timezone id * @param fallback alternative timezone id * @return timezone data */ public static Timezone of( String tzid, TZID fallback ) { Timezone ret = Timezone.getTZ(null, tzid, false); if (ret == null) { ret = Timezone.getTZ(fallback, false); if (ret == null) { ret = Timezone.ofSystem(); } } return ret; } /** *

Creates a new synthetic timezone based only on given data.

* * @param tzid timezone id * @param history history of offset transitions * @return new instance of timezone data * @throws IllegalArgumentException if a fixed zonal offset is combined * with a non-empty history * @since 2.2 */ /*[deutsch] *

Erzeugt eine neue synthetische Zeitzone basierend nur auf den * angegebenen Daten.

* * @param tzid timezone id * @param history history of offset transitions * @return new instance of timezone data * @throws IllegalArgumentException if a fixed zonal offset is combined * with a non-empty history * @since 2.2 */ public static Timezone of( String tzid, TransitionHistory history ) { return new HistorizedTimezone(resolve(tzid), history); } /** *

Gets the associated timezone identifier.

* * @return timezone id * @see java.util.TimeZone#getID() java.util.TimeZone.getID() */ /*[deutsch] *

Liefert die Zeitzonen-ID.

* * @return timezone id * @see java.util.TimeZone#getID() java.util.TimeZone.getID() */ public abstract TZID getID(); /** *

Calculates the total offset for given global timestamp.

* *

Note: The returned offset has never any subsecond part, normally * not even seconds but full minutes or hours.

* * @param ut unix time * @return total shift in seconds which yields local time if added to unix time * @see java.util.TimeZone#getOffset(long) * java.util.TimeZone.getOffset(long) * @see #getStandardOffset(UnixTime) * @see #getDaylightSavingOffset(UnixTime) */ /*[deutsch] *

Ermittelt die gesamte Zeitzonenverschiebung zum angegebenen Zeitpunkt auf * der UT-Weltzeitlinie in Sekunden.

* *

Hinweis: Die zurückgegebene Verschiebung hat niemals * Subsekundenteile, normalerweise auch nicht Sekundenteile, sondern * nur volle Minuten oder Stunden.

* * @param ut unix time * @return total shift in seconds which yields local time if added to unix time * @see java.util.TimeZone#getOffset(long) * java.util.TimeZone.getOffset(long) * @see #getStandardOffset(UnixTime) * @see #getDaylightSavingOffset(UnixTime) */ public abstract ZonalOffset getOffset(UnixTime ut); /** *

Calculates the standard offset for given global timestamp.

* *

Note: The returned offset has never any subsecond part, normally * not even seconds but full minutes or hours.

* * @param ut unix time * @return standard shift in seconds which yields standard local time if added to unix time * @since 3.2/4.1 */ /*[deutsch] *

Ermittelt die Standard-Zeitzonenverschiebung zum angegebenen Zeitpunkt auf * der UT-Weltzeitlinie in Sekunden.

* *

Hinweis: Die zurückgegebene Verschiebung hat niemals * Subsekundenteile, normalerweise auch nicht Sekundenteile, sondern * nur volle Minuten oder Stunden.

* * @param ut unix time * @return standard shift in seconds which yields standard local time if added to unix time * @since 3.2/4.1 */ public abstract ZonalOffset getStandardOffset(UnixTime ut); /** *

Calculates the dst-offset for given global timestamp.

* *

Note: The returned offset has never any subsecond part, normally * not even seconds but full minutes or hours.

* * @param ut unix time * @return dst-shift in seconds which yields local wall time if added to standard local time * @since 3.2/4.1 */ /*[deutsch] *

Ermittelt die Sommerzeitverschiebung zum angegebenen Zeitpunkt auf * der UT-Weltzeitlinie in Sekunden.

* *

Hinweis: Die zurückgegebene Verschiebung hat niemals * Subsekundenteile, normalerweise auch nicht Sekundenteile, sondern * nur volle Minuten oder Stunden.

* * @param ut unix time * @return dst-shift in seconds which yields local wall time if added to standard local time * @since 3.2/4.1 */ public abstract ZonalOffset getDaylightSavingOffset(UnixTime ut); /** *

Calculates the offset for given local timestamp.

* *

In case of gaps or overlaps, this method uses the * {@link #DEFAULT_CONFLICT_STRATEGY standard strategy} * to get the next defined offset. This behaviour is conform to the * JDK-class {@code java.util.GregorianCalendar}.

* *

Note: The returned offset has never any subsecond part, normally * not even seconds but full minutes or hours.

* * @param localDate local date in timezone * @param localTime local wall time in timezone * @return shift in seconds which yields unix time if subtracted * from local time choosing later offset at gaps or overlaps * @see java.util.TimeZone#getOffset(int, int, int, int, int, int) * java.util.TimeZone.getOffset(int, int, int, int, int, int) */ /*[deutsch] *

Ermittelt die Zeitzonenverschiebung zum angegebenen lokalen * Zeitpunkt in Sekunden.

* *

Als Konfliktstrategie für Lücken oder Überlappungen * auf dem lokalen Zeitstrahl wird die {@link #DEFAULT_CONFLICT_STRATEGY * Standardstrategie} verwendet, den nächstdefinierten Offset zu * ermitteln. Dieses Verhalten ist zur JDK-Klasse * {@code java.util.GregorianCalendar} konform.

* *

Hinweis: Die zurückgegebene Verschiebung hat niemals * Subsekundenteile, normalerweise auch nicht Sekundenteile, sondern * nur volle Minuten oder Stunden.

* * @param localDate local date in timezone * @param localTime local wall time in timezone * @return shift in seconds which yields unix time if subtracted * from local time choosing later offset at gaps or overlaps * @see java.util.TimeZone#getOffset(int, int, int, int, int, int) * java.util.TimeZone.getOffset(int, int, int, int, int, int) */ public abstract ZonalOffset getOffset( GregorianDate localDate, WallTime localTime ); /** *

Evaluates if given local timestamp is invalid due to a gap * on the local timeline.

* *

A typical example is the transition from standard to daylight * saving time because the clock will be manually adjusted such * that the clock is moved forward by usually one hour.

* * @param localDate local date in timezone * @param localTime local wall time in timezone * @return {@code true} if the local time is not defined due to * transition gaps else {@code false} */ /*[deutsch] *

Bestimmt, ob der angegebene lokale Zeitpunkt in eine Lücke * fällt.

* *

Das klassiche Beispiel liegt vor, wenn wegen eines Übergangs * von der Winter- zur Sommerzeit eine bestimmte Uhrzeit nicht existiert, * weil die Uhr vorgestellt wurde.

* * @param localDate local date in timezone * @param localTime local wall time in timezone * @return {@code true} if the local time is not defined due to * transition gaps else {@code false} */ public abstract boolean isInvalid( GregorianDate localDate, WallTime localTime ); /** *

Queries if given global timestamp matches daylight saving time * in this timezone?

* *

The DST correction can be obtained as difference between total * offset and raw offset if the raw offset has not changed yet. * As alternative the DST correction can be obtained by evaluating * the transition offset history.

* * @param ut unix time * @return {@code true} if the argument represents summer time * else {@code false} * @see java.util.TimeZone#inDaylightTime(java.util.Date) * java.util.TimeZone.inDaylightTime(java.util.Date) */ /*[deutsch] *

Herrscht zum angegebenen Zeitpunkt Sommerzeit in der Zeitzone?

* *

Die DST-Korrektur selbst kann als Differenz zwischen dem Gesamt-Offset * und dem Standard-Offset erhalten werden, wenn sich der Standard-Offset * historisch nicht geändert hat. Alternativ und genauer kann die * DST-Korrektur über die Offset-Historie ermittelt werden.

* * @param ut unix time * @return {@code true} if the argument represents summer time * else {@code false} * @see java.util.TimeZone#inDaylightTime(java.util.Date) * java.util.TimeZone.inDaylightTime(java.util.Date) */ public abstract boolean isDaylightSaving(UnixTime ut); /** *

Determines if this timezone has no offset transitions and always * uses a fixed offset.

* * @return {@code true} if there is no transition else {@code false} * @since 1.2.1 */ /*[deutsch] *

Legt fest, ob diese Zeitzone keine Übergänge kennt und * nur einen festen Offset benutzt.

* * @return {@code true} if there is no transition else {@code false} * @since 1.2.1 */ public abstract boolean isFixed(); /** *

Gets the underlying offset transitions and rules if available.

* * @return {@code TransitionHistory} or {@code null} if there is no * better {@code Provider} than {@code java.util.TimeZone} */ /*[deutsch] *

Liefert die zugrundeliegenden Übergänge und Regeln, * falls vorhanden.

* * @return {@code TransitionHistory} or {@code null} if there is no * better {@code Provider} than {@code java.util.TimeZone} */ public abstract TransitionHistory getHistory(); /** *

Describes all registered {@code ZoneProvider}-instances with * name and optionally location and version.

* * @return String */ /*[deutsch] *

Beschreibt alle registrierten {@code ZoneProvider}-Instanzen * mit Namen und optional Ort und Version.

* * @return String */ public static String getProviderInfo() { StringBuilder sb = new StringBuilder(128); sb.append(Timezone.class.getName()); sb.append(":[default-provider="); sb.append(DEFAULT_PROVIDER.getName()); sb.append(", registered={"); for (String key : PROVIDERS.keySet()) { ZoneProvider provider = PROVIDERS.get(key); if (provider != null) { // defensive against parallel threads sb.append("(name="); sb.append(provider.getName()); String location = provider.getLocation(); if (!location.isEmpty()) { sb.append(",location="); sb.append(location); } String version = provider.getVersion(); if (!version.isEmpty()) { sb.append(",version="); sb.append(version); } sb.append(')'); } } sb.append("}]"); return sb.toString(); } /** *

Tries to get the version of given registered zone provider.

* * @param provider name of zone provider * @return String (empty if unknown) * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 */ /*[deutsch] *

Versucht die Version des angegebenen und registrierten * {@code ZoneProvider} zu ermitteln.

* * @param provider name of zone provider * @return String (empty if unknown) * @throws IllegalArgumentException if the provider argument is empty * @since 2.2 */ public static String getVersion(String provider) { ZoneProvider zp = getProvider(provider); return ((zp == null) ? "" : zp.getVersion()); } /** *

Yields the names of all registered * {@code ZoneProvider}-instances.

* * @return unmodifiable list of provider names * @since 2.2 * @see ZoneProvider#getName() */ /*[deutsch] *

Liefert die Namen aller registrierten * {@code ZoneProvider}-Instanzen.

* * @return unmodifiable list of provider names * @since 2.2 * @see ZoneProvider#getName() */ public static Set getRegisteredProviders() { return Collections.unmodifiableSet(PROVIDERS.keySet()); } /** *

Gets the strategy for resolving local timestamps.

* * @return transition strategy for resolving local timestamps * @see #with(TransitionStrategy) */ /*[deutsch] *

Ermittelt die Strategie zur Auflösung von lokalen * Zeitstempeln.

* * @return transition strategy for resolving local timestamps * @see #with(TransitionStrategy) */ public abstract TransitionStrategy getStrategy(); /** *

Creates a copy of this timezone which uses given strategy for * resolving local timestamps.

* *

If this timezone has a fixed offset then the strategy will be * ignored because in this case there can never be a conflict. * Otherwise if there is no public offset transition history then * the only supported strategies are {@link #DEFAULT_CONFLICT_STRATEGY} * and {@link #STRICT_MODE}.

* * @param strategy transition strategy for resolving local timestamps * @return copy of this timezone with given strategy * @throws UnsupportedOperationException if given strategy requires * a transition history and this timezone does not have one */ /*[deutsch] *

Erzeugt eine Kopie dieser Zeitzone, die zur Auflösung von * lokalen Zeitstempeln die angegebene Strategie nutzt.

* *

Hat diese Zeitzone einen festen Offset, wird die Strategie * ignoriert, da hier nie eine Konfliktsituation auftreten kann. * Wenn andererseits eine Zeitzone keine öffentliche Historie * kennt, dann werden nur {@link #DEFAULT_CONFLICT_STRATEGY} und * {@link #STRICT_MODE} unterstützt.

* * @param strategy transition strategy for resolving local timestamps * @return copy of this timezone with given strategy * @throws UnsupportedOperationException if given strategy requires * a transition history and this timezone does not have one */ public abstract Timezone with(TransitionStrategy strategy); /** *

Returns the name of this timezone suitable for presentation to * users in given style and locale.

* *

If the name is not available then this method will yield the * ID of this timezone.

* * @param style name style * @param locale language setting * @return localized timezone name for display purposes * @see java.util.TimeZone#getDisplayName(boolean,int,Locale) * java.util.TimeZone.getDisplayName(boolean,int,Locale) * @see Locale#getDefault() * @see #getID() */ /*[deutsch] *

Liefert den anzuzeigenden Zeitzonennamen.

* *

Ist der Zeitzonenname nicht ermittelbar, wird die ID der Zeitzone * geliefert.

* * @param style name style * @param locale language setting * @return localized timezone name for display purposes * @see java.util.TimeZone#getDisplayName(boolean,int,Locale) * java.util.TimeZone.getDisplayName(boolean,int,Locale) * @see Locale#getDefault() * @see #getID() */ public String getDisplayName( NameStyle style, Locale locale ) { String tzid = this.getID().canonical(); int index = tzid.indexOf('~'); ZoneProvider provider = DEFAULT_PROVIDER; String zoneID = tzid; if (index >= 0) { String pname = tzid.substring(0, index); if (!pname.equals(NAME_DEFAULT)) { provider = PROVIDERS.get(pname); } zoneID = tzid.substring(index + 1); } String name = provider.getDisplayName(zoneID, style, locale); if ( name.isEmpty() && (provider != PLATFORM_PROVIDER) ) { // platform provider never returns empty name unless zoneID is empty return PLATFORM_PROVIDER.getDisplayName(zoneID, style, locale); } return name; } /** *

Registers manually the given zone provider.

* *

Repeated registrations of the same provider are ignored.

* * @param provider custom zone provider to be registered * @return {@code true} if registration was successful else {@code false} * @throws IllegalArgumentException if given {@code ZoneProvider} * refers to default, platform or TZDB-provider by name * @since 2.2 */ /*[deutsch] *

Registriert manuell den angegebenen {@code ZoneProvider}.

* *

Wiederholte Registrierungen des gleichen {@code ZoneProvider} * werden ignoriert.

* * @param provider custom zone provider to be registered * @return {@code true} if registration was successful else {@code false} * @throws IllegalArgumentException if given {@code ZoneProvider} * refers to default, platform or TZDB-provider by name * @since 2.2 */ public static boolean registerProvider(ZoneProvider provider) { String name = provider.getName(); if (name.isEmpty()) { throw new IllegalArgumentException( "Missing name of zone provider."); } else if (name.equals(NAME_TZDB)) { throw new IllegalArgumentException( "TZDB provider cannot be registered after startup."); } else if (name.equals(NAME_JUT)) { throw new IllegalArgumentException( "Platform provider cannot be replaced."); } else if (name.equals(NAME_DEFAULT)) { throw new IllegalArgumentException( "Default zone provider cannot be overridden."); } else if (name.equals(NAME_ZONENAMES)) { throw new IllegalArgumentException( "Reserved zone name provider cannot be replaced."); } boolean inserted = (PROVIDERS.putIfAbsent(name, provider) == null); if (inserted) { zonalKeys = new ZonalKeys(); } return inserted; } /** *

Creates a dump of this timezone and writes it to the given * buffer.

* * @param buffer buffer to write the dump to * @throws IOException in any case of I/O-errors * @since 2.2 */ /*[deutsch] *

Erzeugt eine Textzusammenfassung dieser Instanz und schreibt sie * in den angegebenen Puffer.

* * @param buffer buffer to write the dump to * @throws IOException in any case of I/O-errors * @since 2.2 */ public void dump(Appendable buffer) throws IOException { StringBuilder sb = new StringBuilder(4096); sb.append("Start Of Dump =>").append(NEW_LINE); sb.append("*** Timezone-ID:").append(NEW_LINE); sb.append(">>> ").append(this.getID().canonical()).append(NEW_LINE); if (this.isFixed()) { sb.append("*** Fixed offset:").append(NEW_LINE).append(">>> "); sb.append(this.getHistory().getInitialOffset()).append(NEW_LINE); } else { sb.append("*** Strategy:").append(NEW_LINE); sb.append(">>> ").append(this.getStrategy()).append(NEW_LINE); TransitionHistory history = this.getHistory(); sb.append("*** History:").append(NEW_LINE); if (history == null) { sb.append(">>> Not public!").append(NEW_LINE); } else { history.dump(sb); } } sb.append("<= End Of Dump").append(NEW_LINE); buffer.append(sb.toString()); } private static Timezone getDefaultTZ() { String zoneID = java.util.TimeZone.getDefault().getID(); return Timezone.of(zoneID, ZonalOffset.UTC); } private static Timezone getTZ( TZID tzid, boolean wantsException ) { // Liegt eine feste Verschiebung vor? if (tzid instanceof ZonalOffset) { return ((ZonalOffset) tzid).getModel(); } return Timezone.getTZ(tzid, tzid.canonical(), wantsException); } private static Timezone getTZ( TZID tzid, // optional String zoneID, boolean wantsException ) { // Suche im Cache Timezone tz = null; NamedReference sref = CACHE.get(zoneID); if (sref != null) { tz = sref.get(); if (tz == null) { CACHE.remove(sref.tzid); } } if (tz != null) { return tz; } // ZoneProvider auflösen String providerName = ""; String zoneKey = zoneID; for (int i = 0, n = zoneID.length(); i < n; i++) { if (zoneID.charAt(i) == '~') { providerName = zoneID.substring(0, i); zoneKey = zoneID.substring(i + 1); // maybe empty string break; } } if (zoneKey.isEmpty()) { if (wantsException) { throw new IllegalArgumentException("Timezone key is empty."); } else { return null; } } ZoneProvider provider = DEFAULT_PROVIDER; boolean useDefault = ( providerName.isEmpty() || providerName.equals(NAME_DEFAULT)); if (!useDefault) { provider = PROVIDERS.get(providerName); if (provider == null) { if (wantsException) { String msg; if (providerName.equals(NAME_TZDB)) { msg = "TZDB provider not available: "; } else { msg = "Timezone provider not registered: "; } throw new IllegalArgumentException(msg + zoneID); } else { return null; } } } // enums bevorzugen TZID resolved = tzid; if (resolved == null) { if (useDefault) { TZID result = resolve(zoneKey); if (result instanceof ZonalOffset) { return ((ZonalOffset) result).getModel(); } else { resolved = result; } } else { resolved = new NamedID(zoneID); } } // java.util.TimeZone hat keine öffentliche Historie if (provider == PLATFORM_PROVIDER) { PlatformTimezone test = new PlatformTimezone(resolved, zoneKey); // JDK-Fallback verhindern => tz == null if ( !test.isGMT() || zoneKey.equals("GMT") || zoneKey.startsWith("UT") || zoneKey.equals("Z") ) { tz = test; } } else { // exakte Suche in Historie TransitionHistory history = provider.load(zoneKey); if (history == null) { // Alias-Suche + Fallback-Option tz = Timezone.getZoneByAlias(provider, resolved, zoneKey); } else { tz = new HistorizedTimezone(resolved, history); } } // Ungültige ID? if (tz == null) { if (wantsException) { throw new IllegalArgumentException("Unknown timezone: " + zoneID); } else { return null; } } // bei Bedarf im Cache speichern if (cacheActive) { NamedReference oldRef = CACHE.putIfAbsent( zoneID, new NamedReference(tz, QUEUE) ); if (oldRef == null) { synchronized (Timezone.class) { LAST_USED.addFirst(tz); while (LAST_USED.size() >= softLimit) { LAST_USED.removeLast(); } } } else { Timezone oldZone = oldRef.get(); if (oldZone != null) { tz = oldZone; } } } return tz; } private static Timezone getZoneByAlias( ZoneProvider provider, TZID tzid, String zoneKey ) { TransitionHistory history = null; String alias = zoneKey; Map aliases = provider.getAliases(); while ( (history == null) && ((alias = aliases.get(alias)) != null) ) { history = provider.load(alias); } if (history == null) { String fallback = provider.getFallback(); if (fallback.isEmpty()) { return null; } else if (fallback.equals(provider.getName())) { throw new IllegalArgumentException( "Circular zone provider fallback: " + provider.getName()); } else { return new FallbackTimezone( tzid, Timezone.of(fallback + "~" + zoneKey)); } } else { return new HistorizedTimezone(tzid, history); } } private static TZID resolve(String zoneKey) { // enums bevorzugen TZID resolved = PREDEFINED.get(zoneKey); if (resolved == null) { if (zoneKey.startsWith("GMT")) { zoneKey = "UTC" + zoneKey.substring(3); } resolved = ZonalOffset.parse(zoneKey, false); if (resolved == null) { resolved = new NamedID(zoneKey); } } return resolved; } @SuppressWarnings("unchecked") private static List> loadPredefined( ClassLoader loader, String... names ) throws ClassNotFoundException { List> classes = new ArrayList>(); for (String name : names) { Class clazz = Class.forName("net.time4j.tz.olson." + name, true, loader); if (TZID.class.isAssignableFrom(clazz)) { classes.add((Class) clazz); } } return Collections.unmodifiableList(classes); } private static ZoneProvider getProvider(String provider) { if (provider.isEmpty()) { throw new IllegalArgumentException("Missing zone provider."); } return ( provider.equals(NAME_DEFAULT) ? DEFAULT_PROVIDER : PROVIDERS.get(provider)); } private static ZoneProvider compareTZDB( ZoneProvider provider, ZoneProvider zp ) { String v = provider.getVersion(); if (!v.isEmpty()) { if (v.equals(REPOSITORY_VERSION)) { zp = provider; } else if (REPOSITORY_VERSION == null) { if ( (zp == null) || (v.compareTo(zp.getVersion()) > 0) ) { zp = provider; } else if ( (v.compareTo(zp.getVersion()) == 0) && !provider.getLocation().contains("{java.home}") ) { zp = provider; } } } return zp; } //~ Innere Klassen ---------------------------------------------------- /** *

Offers some static methods for the configuration of the * timezone cache.

*/ /*[deutsch] *

Bietet statische Methoden zum Konfigurieren des * Zeitzonendatenpuffers.

*/ public static class Cache { //~ Konstruktoren ------------------------------------------------- private Cache() { // no instantiation } //~ Methoden ------------------------------------------------------ /** *

Can refresh the timezone cache in case of a dynamic * update of the underlying timezone repository.

* *

First the internal cache will be cleared. Furthermore, * if needed the system timezone will be determined again.

*/ /*[deutsch] *

Erlaubt eine Aktualisierung, wenn sich die Zeitzonendatenbank * geändert hat (dynamic update).

* *

Der interne Cache wird entleert. Auch wird bei Bedarf die * Standard-Zeitzone neu ermittelt.

*/ public static void refresh() { synchronized (Timezone.class) { while (QUEUE.poll() != null) {} LAST_USED.clear(); } zonalKeys = new ZonalKeys(); CACHE.clear(); if (ALLOW_SYSTEM_TZ_OVERRIDE) { currentSystemTZ = Timezone.getDefaultTZ(); } } /** *

Aktivates or deactivates the internal cache.

* *

The timezone cache is active by default. Switching off the cache can * make the performance worse especially if the underlying {@code Provider} * itself has no cache.

* * @param active {@code true} if chache shall be active * else {@code false} */ /*[deutsch] *

Aktiviert oder deaktiviert den internen Cache.

* *

Standardmäßig ist der Cache aktiv. Ein Abschalten des * Cache kann die Performance insbesondere dann verschlechtern, wenn der * zugrundeliegende {@code Provider} selbst keinen Cache hat.

* * @param active {@code true} if chache shall be active * else {@code false} */ public static void setCacheActive(boolean active) { cacheActive = active; if (!active) { CACHE.clear(); } } /** *

Updates the size of the internal timezone cache.

* * @param minimumCacheSize new minimum size of cache * @throws IllegalArgumentException if the argument is negative */ /*[deutsch] *

Konfiguriert die Größe des internen Cache neu.

* * @param minimumCacheSize new minimum size of cache * @throws IllegalArgumentException if the argument is negative */ public static void setMinimumCacheSize(int minimumCacheSize) { if (minimumCacheSize < 0) { throw new IllegalArgumentException( "Negative timezone cache size: " + minimumCacheSize); } NamedReference ref; while ((ref = (NamedReference) QUEUE.poll()) != null) { CACHE.remove(ref.tzid); } synchronized (Timezone.class) { softLimit = minimumCacheSize + 1; int n = LAST_USED.size() - minimumCacheSize; for (int i = 0; i < n; i++) { LAST_USED.removeLast(); } } } } private static class NamedReference extends SoftReference { //~ Instanzvariablen ---------------------------------------------- private final String tzid; //~ Konstruktoren ------------------------------------------------- NamedReference( Timezone tz, ReferenceQueue queue ) { super(tz, queue); this.tzid = tz.getID().canonical(); } } private static class NamedID implements TZID, Serializable { //~ Statische Felder/Initialisierungen ---------------------------- private static final long serialVersionUID = -4889632013137688471L; //~ Instanzvariablen ---------------------------------------------- /** * @serial timezone id */ private final String tzid; //~ Konstruktoren ------------------------------------------------- NamedID(String tzid) { super(); this.tzid = tzid; } //~ Methoden ------------------------------------------------------ @Override public String canonical() { return this.tzid; } @Override public boolean equals(Object obj) { if (obj instanceof NamedID) { NamedID that = (NamedID) obj; return this.tzid.equals(that.tzid); } else { return false; } } @Override public int hashCode() { return this.tzid.hashCode(); } @Override public String toString() { return this.getClass().getName() + "@" + this.tzid; } } private static class ZonalKeys { //~ Instanzvariablen ---------------------------------------------- private final List availables; //~ Konstruktoren ------------------------------------------------- ZonalKeys() { super(); List list = new ArrayList(1024); list.add(ZonalOffset.UTC); for (Map.Entry e : PROVIDERS.entrySet()) { ZoneProvider zp = e.getValue(); if ( (zp == PLATFORM_PROVIDER) && (DEFAULT_PROVIDER != PLATFORM_PROVIDER) ) { continue; } for (String id : zp.getAvailableIDs()) { TZID tzid = resolve(id); // wegen resolve() genügt Vergleich per equals() if (!list.contains(tzid)) { list.add(tzid); } } } Collections.sort(list, ID_COMPARATOR); this.availables = Collections.unmodifiableList(list); } } private static class PlatformTZProvider implements ZoneProvider { //~ Methoden ------------------------------------------------------ @Override public Set getAvailableIDs() { Set ret = new HashSet(); String[] temp = java.util.TimeZone.getAvailableIDs(); ret.addAll(Arrays.asList(temp)); return ret; } @Override public Set getPreferredIDs( Locale locale, boolean smart ) { ZoneProvider zp = ZONENAME_PROVIDER; if (zp == null) { return Collections.emptySet(); } return zp.getPreferredIDs(locale, smart); } @Override public Map getAliases() { return Collections.emptyMap(); // JDK hat eingebaute Alias-Suche! } @Override public String getFallback() { return ""; } @Override public String getName() { return "java.util.TimeZone"; } @Override public String getLocation() { return ""; } @Override public String getVersion() { return ""; } @Override public TransitionHistory load(String zoneID) { return null; // leider keine öffentliche Historie!!! } @Override public String getDisplayName( String tzid, NameStyle style, Locale locale ) { if (locale == null) { throw new NullPointerException("Missing locale."); } else if (tzid.isEmpty()) { return ""; } java.util.TimeZone tz = PlatformTimezone.findZone(tzid); if (tz.getID().equals(tzid)) { return tz.getDisplayName( style.isDaylightSaving(), style.isAbbreviation() ? java.util.TimeZone.SHORT : java.util.TimeZone.LONG, locale ); } return tzid; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy