1330 lines
45 KiB
Java
1330 lines
45 KiB
Java
|
/*******************************************************************************
|
||
|
* Copyright 2010 Cees De Groot, Alex Boisvert, Jan Kotek
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
******************************************************************************/
|
||
|
package org.apache.jdbm;
|
||
|
|
||
|
import java.io.*;
|
||
|
import java.lang.reflect.Array;
|
||
|
import java.math.BigDecimal;
|
||
|
import java.math.BigInteger;
|
||
|
import java.util.*;
|
||
|
|
||
|
import static org.apache.jdbm.SerializationHeader.*;
|
||
|
|
||
|
/**
|
||
|
* Serialization util. It reduces serialized data size for most common java types.
|
||
|
* <p/>
|
||
|
* Common pattern is one byte header which identifies data type, then size is written (if required) and
|
||
|
* data.
|
||
|
* <p/>
|
||
|
* On unknown types normal java serialization is used
|
||
|
* <p/>
|
||
|
* <p/>
|
||
|
* Header byte values bellow 180 are reserved by author for future use. If you want to customize
|
||
|
* this class, use values over 180, to be compatible with future updates.
|
||
|
*
|
||
|
* @author Jan Kotek
|
||
|
*/
|
||
|
@SuppressWarnings("unchecked")
|
||
|
public class Serialization extends SerialClassInfo implements Serializer {
|
||
|
|
||
|
|
||
|
/**
|
||
|
* print statistics to STDOUT
|
||
|
*/
|
||
|
static final boolean DEBUG = false;
|
||
|
|
||
|
|
||
|
static final String UTF8 = "UTF-8";
|
||
|
|
||
|
Serialization(DBAbstract db, long serialClassInfoRecid, ArrayList<ClassInfo> info) throws IOException {
|
||
|
super(db, serialClassInfoRecid,info);
|
||
|
}
|
||
|
|
||
|
public Serialization() {
|
||
|
super(null,0L,new ArrayList<ClassInfo>());
|
||
|
// Add java.lang.Object as registered class
|
||
|
registered.add(new ClassInfo(Object.class.getName(), new FieldInfo[]{},false,false));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Serialize the object into a byte array.
|
||
|
*/
|
||
|
public byte[] serialize(Object obj)
|
||
|
throws IOException {
|
||
|
DataInputOutput ba = new DataInputOutput();
|
||
|
|
||
|
serialize(ba, obj);
|
||
|
|
||
|
return ba.toByteArray();
|
||
|
}
|
||
|
|
||
|
boolean isSerializable(Object obj) {
|
||
|
//TODO suboptimal code
|
||
|
try {
|
||
|
serialize(new DataOutputStream(new ByteArrayOutputStream()), obj);
|
||
|
return true;
|
||
|
} catch (Exception e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
public void serialize(final DataOutput out, final Object obj) throws IOException {
|
||
|
serialize(out, obj, null);
|
||
|
}
|
||
|
|
||
|
|
||
|
public void serialize(final DataOutput out, final Object obj, FastArrayList objectStack) throws IOException {
|
||
|
|
||
|
/**try to find object on stack if it exists*/
|
||
|
if (objectStack != null) {
|
||
|
int indexInObjectStack = objectStack.identityIndexOf(obj);
|
||
|
if (indexInObjectStack != -1) {
|
||
|
//object was already serialized, just write reference to it and return
|
||
|
out.write(OBJECT_STACK);
|
||
|
LongPacker.packInt(out, indexInObjectStack);
|
||
|
return;
|
||
|
}
|
||
|
//add this object to objectStack
|
||
|
objectStack.add(obj);
|
||
|
}
|
||
|
|
||
|
final Class clazz = obj != null ? obj.getClass() : null;
|
||
|
|
||
|
/** first try to serialize object without initializing object stack*/
|
||
|
if (obj == null) {
|
||
|
out.write(NULL);
|
||
|
return;
|
||
|
} else if (clazz == Boolean.class) {
|
||
|
if (((Boolean) obj).booleanValue())
|
||
|
out.write(BOOLEAN_TRUE);
|
||
|
else
|
||
|
out.write(BOOLEAN_FALSE);
|
||
|
return;
|
||
|
} else if (clazz == Integer.class) {
|
||
|
final int val = (Integer) obj;
|
||
|
writeInteger(out, val);
|
||
|
return;
|
||
|
} else if (clazz == Double.class) {
|
||
|
double v = (Double) obj;
|
||
|
if (v == -1d)
|
||
|
out.write(DOUBLE_MINUS_1);
|
||
|
else if (v == 0d)
|
||
|
out.write(DOUBLE_0);
|
||
|
else if (v == 1d)
|
||
|
out.write(DOUBLE_1);
|
||
|
else if (v >= 0 && v <= 255 && (int) v == v) {
|
||
|
out.write(DOUBLE_255);
|
||
|
out.write((int) v);
|
||
|
} else if (v >= Short.MIN_VALUE && v <= Short.MAX_VALUE && (short) v == v) {
|
||
|
out.write(DOUBLE_SHORT);
|
||
|
out.writeShort((int) v);
|
||
|
} else {
|
||
|
out.write(DOUBLE_FULL);
|
||
|
out.writeDouble(v);
|
||
|
}
|
||
|
return;
|
||
|
} else if (clazz == Float.class) {
|
||
|
float v = (Float) obj;
|
||
|
if (v == -1f)
|
||
|
out.write(FLOAT_MINUS_1);
|
||
|
else if (v == 0f)
|
||
|
out.write(FLOAT_0);
|
||
|
else if (v == 1f)
|
||
|
out.write(FLOAT_1);
|
||
|
else if (v >= 0 && v <= 255 && (int) v == v) {
|
||
|
out.write(FLOAT_255);
|
||
|
out.write((int) v);
|
||
|
} else if (v >= Short.MIN_VALUE && v <= Short.MAX_VALUE && (short) v == v) {
|
||
|
out.write(FLOAT_SHORT);
|
||
|
out.writeShort((int) v);
|
||
|
|
||
|
} else {
|
||
|
out.write(FLOAT_FULL);
|
||
|
out.writeFloat(v);
|
||
|
}
|
||
|
return;
|
||
|
} else if (clazz == BigInteger.class) {
|
||
|
out.write(BIGINTEGER);
|
||
|
byte[] buf = ((BigInteger) obj).toByteArray();
|
||
|
serializeByteArrayInt(out, buf);
|
||
|
return;
|
||
|
} else if (clazz == BigDecimal.class) {
|
||
|
out.write(BIGDECIMAL);
|
||
|
BigDecimal d = (BigDecimal) obj;
|
||
|
serializeByteArrayInt(out, d.unscaledValue().toByteArray());
|
||
|
LongPacker.packInt(out, d.scale());
|
||
|
return;
|
||
|
} else if (clazz == Long.class) {
|
||
|
final long val = (Long) obj;
|
||
|
writeLong(out, val);
|
||
|
return;
|
||
|
} else if (clazz == Short.class) {
|
||
|
short val = (Short) obj;
|
||
|
if (val == -1)
|
||
|
out.write(SHORT_MINUS_1);
|
||
|
else if (val == 0)
|
||
|
out.write(SHORT_0);
|
||
|
else if (val == 1)
|
||
|
out.write(SHORT_1);
|
||
|
else if (val > 0 && val < 255) {
|
||
|
out.write(SHORT_255);
|
||
|
out.write(val);
|
||
|
} else {
|
||
|
out.write(SHORT_FULL);
|
||
|
out.writeShort(val);
|
||
|
}
|
||
|
return;
|
||
|
} else if (clazz == Byte.class) {
|
||
|
byte val = (Byte) obj;
|
||
|
if (val == -1)
|
||
|
out.write(BYTE_MINUS_1);
|
||
|
else if (val == 0)
|
||
|
out.write(BYTE_0);
|
||
|
else if (val == 1)
|
||
|
out.write(BYTE_1);
|
||
|
else {
|
||
|
out.write(SHORT_FULL);
|
||
|
out.writeByte(val);
|
||
|
}
|
||
|
return;
|
||
|
} else if (clazz == Character.class) {
|
||
|
out.write(CHAR);
|
||
|
out.writeChar((Character) obj);
|
||
|
return;
|
||
|
} else if (clazz == String.class) {
|
||
|
String s = (String) obj;
|
||
|
if (s.length() == 0) {
|
||
|
out.write(STRING_EMPTY);
|
||
|
} else {
|
||
|
out.write(STRING);
|
||
|
serializeString(out, s);
|
||
|
}
|
||
|
return;
|
||
|
} else if (obj instanceof Class) {
|
||
|
out.write(CLASS);
|
||
|
serialize(out, ((Class) obj).getName());
|
||
|
return;
|
||
|
} else if (obj instanceof int[]) {
|
||
|
writeIntArray(out, (int[]) obj);
|
||
|
return;
|
||
|
} else if (obj instanceof long[]) {
|
||
|
writeLongArray(out, (long[]) obj);
|
||
|
return;
|
||
|
} else if (obj instanceof short[]) {
|
||
|
out.write(SHORT_ARRAY);
|
||
|
short[] a = (short[]) obj;
|
||
|
LongPacker.packInt(out,a.length);
|
||
|
for(short s:a) out.writeShort(s);
|
||
|
return;
|
||
|
} else if (obj instanceof boolean[]) {
|
||
|
out.write(BOOLEAN_ARRAY);
|
||
|
boolean[] a = (boolean[]) obj;
|
||
|
LongPacker.packInt(out,a.length);
|
||
|
for(boolean s:a) out.writeBoolean(s); //TODO pack 8 booleans to single byte
|
||
|
return;
|
||
|
} else if (obj instanceof double[]) {
|
||
|
out.write(DOUBLE_ARRAY);
|
||
|
double[] a = (double[]) obj;
|
||
|
LongPacker.packInt(out,a.length);
|
||
|
for(double s:a) out.writeDouble(s);
|
||
|
return;
|
||
|
} else if (obj instanceof float[]) {
|
||
|
out.write(FLOAT_ARRAY);
|
||
|
float[] a = (float[]) obj;
|
||
|
LongPacker.packInt(out,a.length);
|
||
|
for(float s:a) out.writeFloat(s);
|
||
|
return;
|
||
|
} else if (obj instanceof char[]) {
|
||
|
out.write(CHAR_ARRAY);
|
||
|
char[] a = (char[]) obj;
|
||
|
LongPacker.packInt(out,a.length);
|
||
|
for(char s:a) out.writeChar(s);
|
||
|
return;
|
||
|
} else if (obj instanceof byte[]) {
|
||
|
byte[] b = (byte[]) obj;
|
||
|
out.write(ARRAY_BYTE_INT);
|
||
|
serializeByteArrayInt(out, b);
|
||
|
return;
|
||
|
} else if (clazz == Date.class) {
|
||
|
out.write(DATE);
|
||
|
out.writeLong(((Date) obj).getTime());
|
||
|
return;
|
||
|
} else if (clazz == UUID.class) {
|
||
|
out.write(UUID);
|
||
|
serializeUUID(out,(UUID) obj);
|
||
|
return;
|
||
|
} else if (clazz == BTree.class) {
|
||
|
out.write(BTREE);
|
||
|
((BTree) obj).writeExternal(out);
|
||
|
return;
|
||
|
} else if (clazz == HTree.class) {
|
||
|
out.write(HTREE);
|
||
|
((HTree) obj).serialize(out);
|
||
|
return;
|
||
|
} else if (clazz == LinkedList2.class) {
|
||
|
out.write(JDBMLINKEDLIST);
|
||
|
((LinkedList2) obj).serialize(out);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
/** classes bellow need object stack, so initialize it if not alredy initialized*/
|
||
|
if (objectStack == null) {
|
||
|
objectStack = new FastArrayList();
|
||
|
objectStack.add(obj);
|
||
|
}
|
||
|
|
||
|
|
||
|
if (obj instanceof Object[]) {
|
||
|
Object[] b = (Object[]) obj;
|
||
|
boolean packableLongs = b.length <= 255;
|
||
|
if (packableLongs) {
|
||
|
//check if it contains packable longs
|
||
|
for (Object o : b) {
|
||
|
if (o != null && (o.getClass() != Long.class || (((Long) o).longValue() < 0 && ((Long) o).longValue() != Long.MAX_VALUE))) {
|
||
|
packableLongs = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (packableLongs) {
|
||
|
//packable Longs is special case, it is often used in JDBM to reference fields
|
||
|
out.write(ARRAY_OBJECT_PACKED_LONG);
|
||
|
out.write(b.length);
|
||
|
for (Object o : b) {
|
||
|
if (o == null)
|
||
|
LongPacker.packLong(out, 0);
|
||
|
else
|
||
|
LongPacker.packLong(out, ((Long) o).longValue() + 1);
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
out.write(ARRAY_OBJECT);
|
||
|
LongPacker.packInt(out, b.length);
|
||
|
|
||
|
// Write class id for components
|
||
|
Class<?> componentType = obj.getClass().getComponentType();
|
||
|
registerClass(componentType);
|
||
|
//write class header
|
||
|
int classId = getClassId(componentType);
|
||
|
LongPacker.packInt(out, classId);
|
||
|
|
||
|
for (Object o : b)
|
||
|
serialize(out, o, objectStack);
|
||
|
|
||
|
}
|
||
|
|
||
|
} else if (clazz == ArrayList.class) {
|
||
|
ArrayList l = (ArrayList) obj;
|
||
|
boolean packableLongs = l.size() < 255;
|
||
|
if (packableLongs) {
|
||
|
//packable Longs is special case, it is often used in JDBM to reference fields
|
||
|
for (Object o : l) {
|
||
|
if (o != null && (o.getClass() != Long.class || (((Long) o).longValue() < 0 && ((Long) o).longValue() != Long.MAX_VALUE))) {
|
||
|
packableLongs = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (packableLongs) {
|
||
|
out.write(ARRAYLIST_PACKED_LONG);
|
||
|
out.write(l.size());
|
||
|
for (Object o : l) {
|
||
|
if (o == null)
|
||
|
LongPacker.packLong(out, 0);
|
||
|
else
|
||
|
LongPacker.packLong(out, ((Long) o).longValue() + 1);
|
||
|
}
|
||
|
} else {
|
||
|
serializeCollection(ARRAYLIST, out, obj, objectStack);
|
||
|
}
|
||
|
|
||
|
} else if (clazz == java.util.LinkedList.class) {
|
||
|
serializeCollection(LINKEDLIST, out, obj, objectStack);
|
||
|
} else if (clazz == Vector.class) {
|
||
|
serializeCollection(VECTOR, out, obj, objectStack);
|
||
|
} else if (clazz == TreeSet.class) {
|
||
|
TreeSet l = (TreeSet) obj;
|
||
|
out.write(TREESET);
|
||
|
LongPacker.packInt(out, l.size());
|
||
|
serialize(out, l.comparator(), objectStack);
|
||
|
for (Object o : l)
|
||
|
serialize(out, o, objectStack);
|
||
|
} else if (clazz == HashSet.class) {
|
||
|
serializeCollection(HASHSET, out, obj, objectStack);
|
||
|
} else if (clazz == LinkedHashSet.class) {
|
||
|
serializeCollection(LINKEDHASHSET, out, obj, objectStack);
|
||
|
} else if (clazz == TreeMap.class) {
|
||
|
TreeMap l = (TreeMap) obj;
|
||
|
out.write(TREEMAP);
|
||
|
LongPacker.packInt(out, l.size());
|
||
|
serialize(out, l.comparator(), objectStack);
|
||
|
for (Object o : l.keySet()) {
|
||
|
serialize(out, o, objectStack);
|
||
|
serialize(out, l.get(o), objectStack);
|
||
|
}
|
||
|
} else if (clazz == HashMap.class) {
|
||
|
serializeMap(HASHMAP, out, obj, objectStack);
|
||
|
} else if (clazz == IdentityHashMap.class) {
|
||
|
serializeMap(IDENTITYHASHMAP, out, obj, objectStack);
|
||
|
} else if (clazz == LinkedHashMap.class) {
|
||
|
serializeMap(LINKEDHASHMAP, out, obj, objectStack);
|
||
|
} else if (clazz == Hashtable.class) {
|
||
|
serializeMap(HASHTABLE, out, obj, objectStack);
|
||
|
} else if (clazz == Properties.class) {
|
||
|
serializeMap(PROPERTIES, out, obj, objectStack);
|
||
|
} else if (clazz == Locale.class){
|
||
|
out.write(LOCALE);
|
||
|
Locale l = (Locale) obj;
|
||
|
out.writeUTF(l.getLanguage());
|
||
|
out.writeUTF(l.getCountry());
|
||
|
out.writeUTF(l.getVariant());
|
||
|
} else {
|
||
|
out.write(NORMAL);
|
||
|
writeObject(out, obj, objectStack);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
static void serializeString(DataOutput out, String obj) throws IOException {
|
||
|
final int len = obj.length();
|
||
|
LongPacker.packInt(out, len);
|
||
|
for (int i = 0; i < len; i++) {
|
||
|
int c = (int) obj.charAt(i); //TODO investigate if c could be negative here
|
||
|
LongPacker.packInt(out, c);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
private void serializeUUID(DataOutput out, UUID uuid) throws IOException
|
||
|
{
|
||
|
out.writeLong(uuid.getMostSignificantBits());
|
||
|
out.writeLong(uuid.getLeastSignificantBits());
|
||
|
}
|
||
|
|
||
|
private void serializeMap(int header, DataOutput out, Object obj, FastArrayList objectStack) throws IOException {
|
||
|
Map l = (Map) obj;
|
||
|
out.write(header);
|
||
|
LongPacker.packInt(out, l.size());
|
||
|
for (Object o : l.keySet()) {
|
||
|
serialize(out, o, objectStack);
|
||
|
serialize(out, l.get(o), objectStack);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void serializeCollection(int header, DataOutput out, Object obj, FastArrayList objectStack) throws IOException {
|
||
|
Collection l = (Collection) obj;
|
||
|
out.write(header);
|
||
|
LongPacker.packInt(out, l.size());
|
||
|
|
||
|
for (Object o : l)
|
||
|
serialize(out, o, objectStack);
|
||
|
|
||
|
}
|
||
|
|
||
|
private void serializeByteArrayInt(DataOutput out, byte[] b) throws IOException {
|
||
|
LongPacker.packInt(out, b.length);
|
||
|
out.write(b);
|
||
|
}
|
||
|
|
||
|
private void writeLongArray(DataOutput da, long[] obj) throws IOException {
|
||
|
long max = Long.MIN_VALUE;
|
||
|
long min = Long.MAX_VALUE;
|
||
|
for (long i : obj) {
|
||
|
max = Math.max(max, i);
|
||
|
min = Math.min(min, i);
|
||
|
}
|
||
|
|
||
|
if (0 <= min && max <= 255) {
|
||
|
da.write(ARRAY_LONG_B);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (long l : obj)
|
||
|
da.write((int) l);
|
||
|
} else if (0 <= min && max <= Long.MAX_VALUE) {
|
||
|
da.write(ARRAY_LONG_PACKED);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (long l : obj)
|
||
|
LongPacker.packLong(da, l);
|
||
|
} else if (Short.MIN_VALUE <= min && max <= Short.MAX_VALUE) {
|
||
|
da.write(ARRAY_LONG_S);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (long l : obj)
|
||
|
da.writeShort((short) l);
|
||
|
} else if (Integer.MIN_VALUE <= min && max <= Integer.MAX_VALUE) {
|
||
|
da.write(ARRAY_LONG_I);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (long l : obj)
|
||
|
da.writeInt((int) l);
|
||
|
} else {
|
||
|
da.write(ARRAY_LONG_L);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (long l : obj)
|
||
|
da.writeLong(l);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
private void writeIntArray(DataOutput da, int[] obj) throws IOException {
|
||
|
int max = Integer.MIN_VALUE;
|
||
|
int min = Integer.MAX_VALUE;
|
||
|
for (int i : obj) {
|
||
|
max = Math.max(max, i);
|
||
|
min = Math.min(min, i);
|
||
|
}
|
||
|
|
||
|
boolean fitsInByte = 0 <= min && max <= 255;
|
||
|
boolean fitsInShort = Short.MIN_VALUE >= min && max <= Short.MAX_VALUE;
|
||
|
|
||
|
|
||
|
if (obj.length <= 255 && fitsInByte) {
|
||
|
da.write(ARRAY_INT_B_255);
|
||
|
da.write(obj.length);
|
||
|
for (int i : obj)
|
||
|
da.write(i);
|
||
|
} else if (fitsInByte) {
|
||
|
da.write(ARRAY_INT_B_INT);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (int i : obj)
|
||
|
da.write(i);
|
||
|
} else if (0 <= min && max <= Integer.MAX_VALUE) {
|
||
|
da.write(ARRAY_INT_PACKED);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (int l : obj)
|
||
|
LongPacker.packInt(da, l);
|
||
|
} else if (fitsInShort) {
|
||
|
da.write(ARRAY_INT_S);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (int i : obj)
|
||
|
da.writeShort(i);
|
||
|
} else {
|
||
|
da.write(ARRAY_INT_I);
|
||
|
LongPacker.packInt(da, obj.length);
|
||
|
for (int i : obj)
|
||
|
da.writeInt(i);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
private void writeInteger(DataOutput da, final int val) throws IOException {
|
||
|
if (val == -1)
|
||
|
da.write(INTEGER_MINUS_1);
|
||
|
else if (val == 0)
|
||
|
da.write(INTEGER_0);
|
||
|
else if (val == 1)
|
||
|
da.write(INTEGER_1);
|
||
|
else if (val == 2)
|
||
|
da.write(INTEGER_2);
|
||
|
else if (val == 3)
|
||
|
da.write(INTEGER_3);
|
||
|
else if (val == 4)
|
||
|
da.write(INTEGER_4);
|
||
|
else if (val == 5)
|
||
|
da.write(INTEGER_5);
|
||
|
else if (val == 6)
|
||
|
da.write(INTEGER_6);
|
||
|
else if (val == 7)
|
||
|
da.write(INTEGER_7);
|
||
|
else if (val == 8)
|
||
|
da.write(INTEGER_8);
|
||
|
else if (val == Integer.MIN_VALUE)
|
||
|
da.write(INTEGER_MINUS_MAX);
|
||
|
else if (val > 0 && val < 255) {
|
||
|
da.write(INTEGER_255);
|
||
|
da.write(val);
|
||
|
} else if (val < 0) {
|
||
|
da.write(INTEGER_PACK_NEG);
|
||
|
LongPacker.packInt(da, -val);
|
||
|
} else {
|
||
|
da.write(INTEGER_PACK);
|
||
|
LongPacker.packInt(da, val);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void writeLong(DataOutput da, final long val) throws IOException {
|
||
|
if (val == -1)
|
||
|
da.write(LONG_MINUS_1);
|
||
|
else if (val == 0)
|
||
|
da.write(LONG_0);
|
||
|
else if (val == 1)
|
||
|
da.write(LONG_1);
|
||
|
else if (val == 2)
|
||
|
da.write(LONG_2);
|
||
|
else if (val == 3)
|
||
|
da.write(LONG_3);
|
||
|
else if (val == 4)
|
||
|
da.write(LONG_4);
|
||
|
else if (val == 5)
|
||
|
da.write(LONG_5);
|
||
|
else if (val == 6)
|
||
|
da.write(LONG_6);
|
||
|
else if (val == 7)
|
||
|
da.write(LONG_7);
|
||
|
else if (val == 8)
|
||
|
da.write(LONG_8);
|
||
|
else if (val == Long.MIN_VALUE)
|
||
|
da.write(LONG_MINUS_MAX);
|
||
|
else if (val > 0 && val < 255) {
|
||
|
da.write(LONG_255);
|
||
|
da.write((int) val);
|
||
|
} else if (val < 0) {
|
||
|
da.write(LONG_PACK_NEG);
|
||
|
LongPacker.packLong(da, -val);
|
||
|
} else {
|
||
|
da.write(LONG_PACK);
|
||
|
LongPacker.packLong(da, val);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Deserialize an object from a byte array
|
||
|
*
|
||
|
* @throws IOException
|
||
|
* @throws ClassNotFoundException
|
||
|
*/
|
||
|
public Object deserialize(byte[] buf) throws ClassNotFoundException, IOException {
|
||
|
DataInputOutput bs = new DataInputOutput(buf);
|
||
|
Object ret = deserialize(bs);
|
||
|
if (bs.available() != 0)
|
||
|
throw new InternalError("bytes left: " + bs.available());
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
static String deserializeString(DataInput buf) throws IOException {
|
||
|
int len = LongPacker.unpackInt(buf);
|
||
|
char[] b = new char[len];
|
||
|
for (int i = 0; i < len; i++)
|
||
|
b[i] = (char) LongPacker.unpackInt(buf);
|
||
|
|
||
|
return new String(b);
|
||
|
}
|
||
|
|
||
|
|
||
|
public Object deserialize(DataInput is) throws IOException, ClassNotFoundException {
|
||
|
return deserialize(is, null);
|
||
|
}
|
||
|
|
||
|
public Object deserialize(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
|
||
|
Object ret = null;
|
||
|
|
||
|
final int head = is.readUnsignedByte();
|
||
|
|
||
|
/** first try to deserialize object without allocating object stack*/
|
||
|
switch (head) {
|
||
|
case NULL:
|
||
|
break;
|
||
|
case BOOLEAN_TRUE:
|
||
|
ret = Boolean.TRUE;
|
||
|
break;
|
||
|
case BOOLEAN_FALSE:
|
||
|
ret = Boolean.FALSE;
|
||
|
break;
|
||
|
case INTEGER_MINUS_1:
|
||
|
ret = Integer.valueOf(-1);
|
||
|
break;
|
||
|
case INTEGER_0:
|
||
|
ret = Integer.valueOf(0);
|
||
|
break;
|
||
|
case INTEGER_1:
|
||
|
ret = Integer.valueOf(1);
|
||
|
break;
|
||
|
case INTEGER_2:
|
||
|
ret = Integer.valueOf(2);
|
||
|
break;
|
||
|
case INTEGER_3:
|
||
|
ret = Integer.valueOf(3);
|
||
|
break;
|
||
|
case INTEGER_4:
|
||
|
ret = Integer.valueOf(4);
|
||
|
break;
|
||
|
case INTEGER_5:
|
||
|
ret = Integer.valueOf(5);
|
||
|
break;
|
||
|
case INTEGER_6:
|
||
|
ret = Integer.valueOf(6);
|
||
|
break;
|
||
|
case INTEGER_7:
|
||
|
ret = Integer.valueOf(7);
|
||
|
break;
|
||
|
case INTEGER_8:
|
||
|
ret = Integer.valueOf(8);
|
||
|
break;
|
||
|
case INTEGER_MINUS_MAX:
|
||
|
ret = Integer.valueOf(Integer.MIN_VALUE);
|
||
|
break;
|
||
|
case INTEGER_255:
|
||
|
ret = Integer.valueOf(is.readUnsignedByte());
|
||
|
break;
|
||
|
case INTEGER_PACK_NEG:
|
||
|
ret = Integer.valueOf(-LongPacker.unpackInt(is));
|
||
|
break;
|
||
|
case INTEGER_PACK:
|
||
|
ret = Integer.valueOf(LongPacker.unpackInt(is));
|
||
|
break;
|
||
|
case LONG_MINUS_1:
|
||
|
ret = Long.valueOf(-1);
|
||
|
break;
|
||
|
case LONG_0:
|
||
|
ret = Long.valueOf(0);
|
||
|
break;
|
||
|
case LONG_1:
|
||
|
ret = Long.valueOf(1);
|
||
|
break;
|
||
|
case LONG_2:
|
||
|
ret = Long.valueOf(2);
|
||
|
break;
|
||
|
case LONG_3:
|
||
|
ret = Long.valueOf(3);
|
||
|
break;
|
||
|
case LONG_4:
|
||
|
ret = Long.valueOf(4);
|
||
|
break;
|
||
|
case LONG_5:
|
||
|
ret = Long.valueOf(5);
|
||
|
break;
|
||
|
case LONG_6:
|
||
|
ret = Long.valueOf(6);
|
||
|
break;
|
||
|
case LONG_7:
|
||
|
ret = Long.valueOf(7);
|
||
|
break;
|
||
|
case LONG_8:
|
||
|
ret = Long.valueOf(8);
|
||
|
break;
|
||
|
case LONG_255:
|
||
|
ret = Long.valueOf(is.readUnsignedByte());
|
||
|
break;
|
||
|
case LONG_PACK_NEG:
|
||
|
ret = Long.valueOf(-LongPacker.unpackLong(is));
|
||
|
break;
|
||
|
case LONG_PACK:
|
||
|
ret = Long.valueOf(LongPacker.unpackLong(is));
|
||
|
break;
|
||
|
case LONG_MINUS_MAX:
|
||
|
ret = Long.valueOf(Long.MIN_VALUE);
|
||
|
break;
|
||
|
case SHORT_MINUS_1:
|
||
|
ret = Short.valueOf((short) -1);
|
||
|
break;
|
||
|
case SHORT_0:
|
||
|
ret = Short.valueOf((short) 0);
|
||
|
break;
|
||
|
case SHORT_1:
|
||
|
ret = Short.valueOf((short) 1);
|
||
|
break;
|
||
|
case SHORT_255:
|
||
|
ret = Short.valueOf((short) is.readUnsignedByte());
|
||
|
break;
|
||
|
case SHORT_FULL:
|
||
|
ret = Short.valueOf(is.readShort());
|
||
|
break;
|
||
|
case BYTE_MINUS_1:
|
||
|
ret = Byte.valueOf((byte) -1);
|
||
|
break;
|
||
|
case BYTE_0:
|
||
|
ret = Byte.valueOf((byte) 0);
|
||
|
break;
|
||
|
case BYTE_1:
|
||
|
ret = Byte.valueOf((byte) 1);
|
||
|
break;
|
||
|
case BYTE_FULL:
|
||
|
ret = Byte.valueOf(is.readByte());
|
||
|
break;
|
||
|
case SHORT_ARRAY:
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
ret = new short[size];
|
||
|
for(int i=0;i<size;i++) ((short[])ret)[i] = is.readShort();
|
||
|
break;
|
||
|
case BOOLEAN_ARRAY:
|
||
|
size = LongPacker.unpackInt(is);
|
||
|
ret = new boolean[size];
|
||
|
for(int i=0;i<size;i++) ((boolean[])ret)[i] = is.readBoolean();
|
||
|
break;
|
||
|
case DOUBLE_ARRAY:
|
||
|
size = LongPacker.unpackInt(is);
|
||
|
ret = new double[size];
|
||
|
for(int i=0;i<size;i++) ((double[])ret)[i] = is.readDouble();
|
||
|
break;
|
||
|
case FLOAT_ARRAY:
|
||
|
size = LongPacker.unpackInt(is);
|
||
|
ret = new float[size];
|
||
|
for(int i=0;i<size;i++) ((float[])ret)[i] = is.readFloat();
|
||
|
break;
|
||
|
case CHAR_ARRAY:
|
||
|
size = LongPacker.unpackInt(is);
|
||
|
ret = new char[size];
|
||
|
for(int i=0;i<size;i++) ((char[])ret)[i] = is.readChar();
|
||
|
break;
|
||
|
case CHAR:
|
||
|
ret = Character.valueOf(is.readChar());
|
||
|
break;
|
||
|
case FLOAT_MINUS_1:
|
||
|
ret = Float.valueOf(-1);
|
||
|
break;
|
||
|
case FLOAT_0:
|
||
|
ret = Float.valueOf(0);
|
||
|
break;
|
||
|
case FLOAT_1:
|
||
|
ret = Float.valueOf(1);
|
||
|
break;
|
||
|
case FLOAT_255:
|
||
|
ret = Float.valueOf(is.readUnsignedByte());
|
||
|
break;
|
||
|
case FLOAT_SHORT:
|
||
|
ret = Float.valueOf(is.readShort());
|
||
|
break;
|
||
|
case FLOAT_FULL:
|
||
|
ret = Float.valueOf(is.readFloat());
|
||
|
break;
|
||
|
case DOUBLE_MINUS_1:
|
||
|
ret = Double.valueOf(-1);
|
||
|
break;
|
||
|
case DOUBLE_0:
|
||
|
ret = Double.valueOf(0);
|
||
|
break;
|
||
|
case DOUBLE_1:
|
||
|
ret = Double.valueOf(1);
|
||
|
break;
|
||
|
case DOUBLE_255:
|
||
|
ret = Double.valueOf(is.readUnsignedByte());
|
||
|
break;
|
||
|
case DOUBLE_SHORT:
|
||
|
ret = Double.valueOf(is.readShort());
|
||
|
break;
|
||
|
case DOUBLE_FULL:
|
||
|
ret = Double.valueOf(is.readDouble());
|
||
|
break;
|
||
|
case BIGINTEGER:
|
||
|
ret = new BigInteger(deserializeArrayByteInt(is));
|
||
|
break;
|
||
|
case BIGDECIMAL:
|
||
|
ret = new BigDecimal(new BigInteger(deserializeArrayByteInt(is)), LongPacker.unpackInt(is));
|
||
|
break;
|
||
|
case STRING:
|
||
|
ret = deserializeString(is);
|
||
|
break;
|
||
|
case STRING_EMPTY:
|
||
|
ret = Utils.EMPTY_STRING;
|
||
|
break;
|
||
|
|
||
|
case CLASS:
|
||
|
ret = deserializeClass(is);
|
||
|
break;
|
||
|
case DATE:
|
||
|
ret = new Date(is.readLong());
|
||
|
break;
|
||
|
case UUID:
|
||
|
ret = deserializeUUID(is);
|
||
|
break;
|
||
|
case ARRAY_INT_B_255:
|
||
|
ret = deserializeArrayIntB255(is);
|
||
|
break;
|
||
|
case ARRAY_INT_B_INT:
|
||
|
ret = deserializeArrayIntBInt(is);
|
||
|
break;
|
||
|
case ARRAY_INT_S:
|
||
|
ret = deserializeArrayIntSInt(is);
|
||
|
break;
|
||
|
case ARRAY_INT_I:
|
||
|
ret = deserializeArrayIntIInt(is);
|
||
|
break;
|
||
|
case ARRAY_INT_PACKED:
|
||
|
ret = deserializeArrayIntPack(is);
|
||
|
break;
|
||
|
case ARRAY_LONG_B:
|
||
|
ret = deserializeArrayLongB(is);
|
||
|
break;
|
||
|
case ARRAY_LONG_S:
|
||
|
ret = deserializeArrayLongS(is);
|
||
|
break;
|
||
|
case ARRAY_LONG_I:
|
||
|
ret = deserializeArrayLongI(is);
|
||
|
break;
|
||
|
case ARRAY_LONG_L:
|
||
|
ret = deserializeArrayLongL(is);
|
||
|
break;
|
||
|
case ARRAY_LONG_PACKED:
|
||
|
ret = deserializeArrayLongPack(is);
|
||
|
break;
|
||
|
case ARRAYLIST_PACKED_LONG:
|
||
|
ret = deserializeArrayListPackedLong(is);
|
||
|
break;
|
||
|
case ARRAY_BYTE_INT:
|
||
|
ret = deserializeArrayByteInt(is);
|
||
|
break;
|
||
|
case LOCALE :
|
||
|
ret = new Locale(is.readUTF(),is.readUTF(),is.readUTF());
|
||
|
break;
|
||
|
case JDBMLINKEDLIST:
|
||
|
ret = LinkedList2.deserialize(is, this);
|
||
|
break;
|
||
|
case HTREE:
|
||
|
ret = HTree.deserialize(is,this);
|
||
|
break;
|
||
|
case BTREE:
|
||
|
ret = BTree.readExternal(is,this);
|
||
|
break;
|
||
|
case BTREE_NODE_LEAF:
|
||
|
throw new InternalError("BPage header, wrong serializer used");
|
||
|
case BTREE_NODE_NONLEAF:
|
||
|
throw new InternalError("BPage header, wrong serializer used");
|
||
|
case JAVA_SERIALIZATION:
|
||
|
throw new InternalError("Wrong header, data were probably serialized with OutputStream, not with JDBM serialization");
|
||
|
|
||
|
case -1:
|
||
|
throw new EOFException();
|
||
|
|
||
|
}
|
||
|
|
||
|
if (ret != null || head == NULL) {
|
||
|
if (objectStack != null)
|
||
|
objectStack.add(ret);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/** something else which needs object stack initialized*/
|
||
|
|
||
|
if (objectStack == null)
|
||
|
objectStack = new FastArrayList();
|
||
|
int oldObjectStackSize = objectStack.size();
|
||
|
|
||
|
switch (head) {
|
||
|
case NORMAL:
|
||
|
ret = readObject(is, objectStack);
|
||
|
break;
|
||
|
case OBJECT_STACK:
|
||
|
ret = objectStack.get(LongPacker.unpackInt(is));
|
||
|
break;
|
||
|
case ARRAYLIST:
|
||
|
ret = deserializeArrayList(is, objectStack);
|
||
|
break;
|
||
|
case ARRAY_OBJECT:
|
||
|
ret = deserializeArrayObject(is, objectStack);
|
||
|
break;
|
||
|
case ARRAY_OBJECT_PACKED_LONG:
|
||
|
ret = deserializeArrayObjectPackedLong(is);
|
||
|
break;
|
||
|
case LINKEDLIST:
|
||
|
ret = deserializeLinkedList(is, objectStack);
|
||
|
break;
|
||
|
case TREESET:
|
||
|
ret = deserializeTreeSet(is, objectStack);
|
||
|
break;
|
||
|
case HASHSET:
|
||
|
ret = deserializeHashSet(is, objectStack);
|
||
|
break;
|
||
|
case LINKEDHASHSET:
|
||
|
ret = deserializeLinkedHashSet(is, objectStack);
|
||
|
break;
|
||
|
case VECTOR:
|
||
|
ret = deserializeVector(is, objectStack);
|
||
|
break;
|
||
|
case TREEMAP:
|
||
|
ret = deserializeTreeMap(is, objectStack);
|
||
|
break;
|
||
|
case HASHMAP:
|
||
|
ret = deserializeHashMap(is, objectStack);
|
||
|
break;
|
||
|
case IDENTITYHASHMAP:
|
||
|
ret = deserializeIdentityHashMap(is, objectStack);
|
||
|
break;
|
||
|
case LINKEDHASHMAP:
|
||
|
ret = deserializeLinkedHashMap(is, objectStack);
|
||
|
break;
|
||
|
case HASHTABLE:
|
||
|
ret = deserializeHashtable(is, objectStack);
|
||
|
break;
|
||
|
case PROPERTIES:
|
||
|
ret = deserializeProperties(is, objectStack);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
throw new InternalError("Unknown serialization header: " + head);
|
||
|
}
|
||
|
|
||
|
if (head != OBJECT_STACK && objectStack.size() == oldObjectStackSize) {
|
||
|
//check if object was not already added to stack as part of collection
|
||
|
objectStack.add(ret);
|
||
|
}
|
||
|
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private Class deserializeClass(DataInput is) throws IOException, ClassNotFoundException {
|
||
|
String className = (String) deserialize(is);
|
||
|
Class cls = Class.forName(className);
|
||
|
return cls;
|
||
|
}
|
||
|
|
||
|
|
||
|
private byte[] deserializeArrayByteInt(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
byte[] b = new byte[size];
|
||
|
is.readFully(b);
|
||
|
return b;
|
||
|
}
|
||
|
|
||
|
|
||
|
private long[] deserializeArrayLongL(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
long[] ret = new long[size];
|
||
|
for (int i = 0; i < size; i++)
|
||
|
ret[i] = is.readLong();
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private long[] deserializeArrayLongI(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
long[] ret = new long[size];
|
||
|
for (int i = 0; i < size; i++)
|
||
|
ret[i] = is.readInt();
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private long[] deserializeArrayLongS(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
long[] ret = new long[size];
|
||
|
for (int i = 0; i < size; i++)
|
||
|
ret[i] = is.readShort();
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private long[] deserializeArrayLongB(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
long[] ret = new long[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
ret[i] = is.readUnsignedByte();
|
||
|
if (ret[i] < 0)
|
||
|
throw new EOFException();
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private int[] deserializeArrayIntIInt(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
int[] ret = new int[size];
|
||
|
for (int i = 0; i < size; i++)
|
||
|
ret[i] = is.readInt();
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private int[] deserializeArrayIntSInt(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
int[] ret = new int[size];
|
||
|
for (int i = 0; i < size; i++)
|
||
|
ret[i] = is.readShort();
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private int[] deserializeArrayIntBInt(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
int[] ret = new int[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
ret[i] = is.readUnsignedByte();
|
||
|
if (ret[i] < 0)
|
||
|
throw new EOFException();
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private int[] deserializeArrayIntPack(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
if (size < 0)
|
||
|
throw new EOFException();
|
||
|
|
||
|
int[] ret = new int[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
ret[i] = LongPacker.unpackInt(is);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
private long[] deserializeArrayLongPack(DataInput is) throws IOException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
if (size < 0)
|
||
|
throw new EOFException();
|
||
|
|
||
|
long[] ret = new long[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
ret[i] = LongPacker.unpackLong(is);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
private UUID deserializeUUID(DataInput is) throws IOException
|
||
|
{
|
||
|
return new UUID(is.readLong(), is.readLong());
|
||
|
}
|
||
|
|
||
|
private int[] deserializeArrayIntB255(DataInput is) throws IOException {
|
||
|
int size = is.readUnsignedByte();
|
||
|
if (size < 0)
|
||
|
throw new EOFException();
|
||
|
|
||
|
int[] ret = new int[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
ret[i] = is.readUnsignedByte();
|
||
|
if (ret[i] < 0)
|
||
|
throw new EOFException();
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
private Object[] deserializeArrayObject(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
// Read class id for components
|
||
|
int classId = LongPacker.unpackInt(is);
|
||
|
Class clazz = classId2class.get(classId);
|
||
|
if (clazz == null) clazz = Object.class;
|
||
|
|
||
|
Object[] s = (Object[])Array.newInstance(clazz, size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s[i] = deserialize(is, objectStack);
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
private Object[] deserializeArrayObjectPackedLong(DataInput is) throws IOException, ClassNotFoundException {
|
||
|
int size = is.readUnsignedByte();
|
||
|
Object[] s = new Object[size];
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
long l = LongPacker.unpackLong(is);
|
||
|
if (l == 0)
|
||
|
s[i] = null;
|
||
|
else
|
||
|
s[i] = Long.valueOf(l - 1);
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private ArrayList<Object> deserializeArrayList(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
ArrayList<Object> s = new ArrayList<Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
private ArrayList<Object> deserializeArrayListPackedLong(DataInput is) throws IOException, ClassNotFoundException {
|
||
|
int size = is.readUnsignedByte();
|
||
|
if (size < 0)
|
||
|
throw new EOFException();
|
||
|
|
||
|
ArrayList<Object> s = new ArrayList<Object>(size);
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
long l = LongPacker.unpackLong(is);
|
||
|
if (l == 0)
|
||
|
s.add(null);
|
||
|
else
|
||
|
s.add(Long.valueOf(l - 1));
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private java.util.LinkedList deserializeLinkedList(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
java.util.LinkedList s = new java.util.LinkedList();
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private Vector<Object> deserializeVector(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
Vector<Object> s = new Vector<Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private HashSet<Object> deserializeHashSet(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
HashSet<Object> s = new HashSet<Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private LinkedHashSet<Object> deserializeLinkedHashSet(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
LinkedHashSet<Object> s = new LinkedHashSet<Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private TreeSet<Object> deserializeTreeSet(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
TreeSet<Object> s = new TreeSet<Object>();
|
||
|
objectStack.add(s);
|
||
|
Comparator comparator = (Comparator) deserialize(is, objectStack);
|
||
|
if (comparator != null)
|
||
|
s = new TreeSet<Object>(comparator);
|
||
|
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.add(deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private TreeMap<Object, Object> deserializeTreeMap(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
TreeMap<Object, Object> s = new TreeMap<Object, Object>();
|
||
|
objectStack.add(s);
|
||
|
Comparator comparator = (Comparator) deserialize(is, objectStack);
|
||
|
if (comparator != null)
|
||
|
s = new TreeMap<Object, Object>(comparator);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private HashMap<Object, Object> deserializeHashMap(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
HashMap<Object, Object> s = new HashMap<Object, Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
private IdentityHashMap<Object, Object> deserializeIdentityHashMap(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
IdentityHashMap<Object, Object> s = new IdentityHashMap<Object, Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
private LinkedHashMap<Object, Object> deserializeLinkedHashMap(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
LinkedHashMap<Object, Object> s = new LinkedHashMap<Object, Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private Hashtable<Object, Object> deserializeHashtable(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
Hashtable<Object, Object> s = new Hashtable<Object, Object>(size);
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
private Properties deserializeProperties(DataInput is, FastArrayList objectStack) throws IOException, ClassNotFoundException {
|
||
|
int size = LongPacker.unpackInt(is);
|
||
|
|
||
|
Properties s = new Properties();
|
||
|
objectStack.add(s);
|
||
|
for (int i = 0; i < size; i++)
|
||
|
s.put(deserialize(is, objectStack), deserialize(is, objectStack));
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Utility class similar to ArrayList, but with fast identity search.
|
||
|
*/
|
||
|
static class FastArrayList<K> {
|
||
|
|
||
|
private int size = 0;
|
||
|
private K[] elementData = (K[]) new Object[8];
|
||
|
|
||
|
K get(int index) {
|
||
|
if (index >= size) throw new IndexOutOfBoundsException();
|
||
|
return elementData[index];
|
||
|
}
|
||
|
|
||
|
void add(K o) {
|
||
|
if (elementData.length == size) {
|
||
|
//grow array if necessary
|
||
|
elementData = Arrays.copyOf(elementData, elementData.length * 2);
|
||
|
}
|
||
|
|
||
|
elementData[size] = o;
|
||
|
size++;
|
||
|
}
|
||
|
|
||
|
int size() {
|
||
|
return size;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* This method is reason why ArrayList is not used.
|
||
|
* Search an item in list and returns its index.
|
||
|
* It uses identity rather than 'equalsTo'
|
||
|
* One could argue that TreeMap should be used instead,
|
||
|
* but we do not expect large object trees.
|
||
|
* This search is VERY FAST compared to Maps, it does not allocate
|
||
|
* new instances or uses method calls.
|
||
|
*
|
||
|
* @param obj
|
||
|
* @return index of object in list or -1 if not found
|
||
|
*/
|
||
|
int identityIndexOf(Object obj) {
|
||
|
for (int i = 0; i < size; i++) {
|
||
|
if (obj == elementData[i])
|
||
|
return i;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
}
|