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

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

/*
 * -----------------------------------------------------------------------
 * Copyright © 2013-2015 Meno Hochschild, 
 * -----------------------------------------------------------------------
 * This file (SPX.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 java.io.Externalizable;
import java.io.IOException;
import java.io.InvalidClassException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.io.ObjectStreamException;
import java.io.StreamCorruptedException;


/**
 * 

Serialization Proxy für eine zonale Verschiebung * oder eine (historisierte) Zeitzone.

* * @author Meno Hochschild * @serial include */ final class SPX implements Externalizable { //~ Statische Felder/Initialisierungen -------------------------------- /** Serialisierungstyp von {@code FallbackTimezone}. */ static final int FALLBACK_TIMEZONE_TYPE = 12; /** Serialisierungstyp von {@code TransitionResolver}. */ static final int TRANSITION_RESOLVER_TYPE = 13; /** Serialisierungstyp von {@code HistorizedTimezone}. */ static final int HISTORIZED_TIMEZONE_TYPE = 14; /** Serialisierungstyp von {@code ZonalOffset}. */ static final int ZONAL_OFFSET_TYPE = 15; private static final long serialVersionUID = -1000776907354520172L; //~ Instanzvariablen -------------------------------------------------- private transient Object obj; private transient int type; //~ Konstruktoren ----------------------------------------------------- /** *

Benutzt in der Deserialisierung gemäß dem Kontrakt * von {@code Externalizable}.

*/ public SPX() { super(); } /** *

Benutzt in der Serialisierung (writeReplace).

* * @param obj object to be serialized * @param type serialization type corresponding to type of obj */ SPX( Object obj, int type ) { super(); this.obj = obj; this.type = type; } //~ Methoden ---------------------------------------------------------- /** *

Implementation method of interface {@link Externalizable}.

* *

The first byte contains within the 4 most-significant bits the type * of the object to be serialized. Then the data bytes follow in a * bit-compressed representation.

* * @serialData data layout see {@code writeReplace()}-method of object * to be serialized * @param out output stream * @throws IOException in case of I/O-problems */ /*[deutsch] *

Implementierungsmethode des Interface {@link Externalizable}.

* *

Das erste Byte enthält um 4 Bits nach links verschoben den * Typ des zu serialisierenden Objekts. Danach folgen die Daten-Bits * in einer bit-komprimierten Darstellung.

* * @serialData data layout see {@code writeReplace()}-method of object * to be serialized * @param out output stream * @throws IOException in case of I/O-problems */ @Override public void writeExternal(ObjectOutput out) throws IOException { switch (this.type) { case FALLBACK_TIMEZONE_TYPE: this.writeFallback(out); break; case TRANSITION_RESOLVER_TYPE: this.writeStrategy(out); break; case HISTORIZED_TIMEZONE_TYPE: this.writeZone(out); break; case ZONAL_OFFSET_TYPE: this.writeOffset(out); break; default: throw new InvalidClassException("Unknown serialized type."); } } /** *

Implementation method of interface {@link Externalizable}.

* * @param in input stream * @throws IOException in case of I/O-problems * @throws ClassNotFoundException if class-loading fails */ /*[deutsch] *

Implementierungsmethode des Interface {@link Externalizable}.

* * @param in input stream * @throws IOException in case of I/O-problems * @throws ClassNotFoundException if class-loading fails */ @Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { byte header = in.readByte(); switch ((header & 0xFF) >> 4) { case FALLBACK_TIMEZONE_TYPE: this.obj = this.readFallback(in, header); break; case TRANSITION_RESOLVER_TYPE: this.obj = this.readStrategy(in, header); break; case HISTORIZED_TIMEZONE_TYPE: this.obj = this.readZone(in, header); break; case ZONAL_OFFSET_TYPE: this.obj = this.readOffset(in, header); break; default: throw new StreamCorruptedException("Unknown serialized type."); } } private Object readResolve() throws ObjectStreamException { return this.obj; } private void writeFallback(ObjectOutput out) throws IOException { FallbackTimezone tz = (FallbackTimezone) this.obj; int header = (FALLBACK_TIMEZONE_TYPE << 4); out.writeByte(header); out.writeObject(tz.getID()); out.writeObject(tz.getFallback()); } private Object readFallback( ObjectInput in, byte header ) throws IOException, ClassNotFoundException { TZID id = (TZID) in.readObject(); Timezone fallback = (Timezone) in.readObject(); return new FallbackTimezone(id, fallback); } private void writeStrategy(ObjectOutput out) throws IOException { TransitionResolver resolver = (TransitionResolver) this.obj; int header = (TRANSITION_RESOLVER_TYPE << 4); header |= resolver.getKey(); out.writeByte(header); } private Object readStrategy( ObjectInput in, byte header ) throws IOException, ClassNotFoundException { int key = (header & 0x0F); GapResolver gapResolver = GapResolver.values()[key / 2]; OverlapResolver overlapResolver = OverlapResolver.values()[key % 2]; return TransitionResolver.of(gapResolver, overlapResolver); } private void writeZone(ObjectOutput out) throws IOException { HistorizedTimezone tz = (HistorizedTimezone) this.obj; int header = (HISTORIZED_TIMEZONE_TYPE << 4); boolean specialStrategy = (tz.getStrategy() != Timezone.DEFAULT_CONFLICT_STRATEGY); if (specialStrategy) { header |= 1; } out.writeByte(header); out.writeObject(tz.getID()); out.writeObject(tz.getHistory()); if (specialStrategy) { out.writeObject(tz.getStrategy()); } } private Object readZone( ObjectInput in, byte header ) throws IOException, ClassNotFoundException { TZID id = (TZID) in.readObject(); TransitionHistory history = (TransitionHistory) in.readObject(); TransitionStrategy strategy = Timezone.DEFAULT_CONFLICT_STRATEGY; if ((header & 0x0F) == 1) { strategy = (TransitionStrategy) in.readObject(); } return new HistorizedTimezone(id, history, strategy); } private void writeOffset(ObjectOutput out) throws IOException { ZonalOffset offset = (ZonalOffset) this.obj; boolean hasFraction = (offset.getFractionalAmount() != 0); int header = (ZONAL_OFFSET_TYPE << 4); if (hasFraction) { header |= 1; } out.writeByte(header); out.writeInt(offset.getIntegralAmount()); if (hasFraction) { out.writeInt(offset.getFractionalAmount()); } } private Object readOffset( ObjectInput in, byte header ) throws IOException { int offset = in.readInt(); int fraction = 0; if ((header & 0x0F) == 1) { fraction = in.readInt(); } return ZonalOffset.ofTotalSeconds(offset, fraction); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy