📄 objectinputstream.java
字号:
/* ObjectInputStream.java -- Class used to read serialized objects Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003 Free Software Foundation, Inc.This file is part of GNU Classpath.GNU Classpath is free software; you can redistribute it and/or modifyit under the terms of the GNU General Public License as published bythe Free Software Foundation; either version 2, or (at your option)any later version.GNU Classpath is distributed in the hope that it will be useful, butWITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNUGeneral Public License for more details.You should have received a copy of the GNU General Public Licensealong with GNU Classpath; see the file COPYING. If not, write to theFree Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA02111-1307 USA.Linking this library statically or dynamically with other modules ismaking a combined work based on this library. Thus, the terms andconditions of the GNU General Public License cover the wholecombination.As a special exception, the copyright holders of this library give youpermission to link this library with independent modules to produce anexecutable, regardless of the license terms of these independentmodules, and to copy and distribute the resulting executable underterms of your choice, provided that you also meet, for each linkedindependent module, the terms and conditions of the license of thatmodule. An independent module is a module which is not derived fromor based on this library. If you modify this library, you may extendthis exception to your version of the library, but you are notobligated to do so. If you do not wish to do so, delete thisexception statement from your version. */package java.io;import java.lang.reflect.Array;import java.lang.reflect.Modifier;import java.lang.reflect.Proxy;import java.util.Arrays;import java.util.Hashtable;import java.util.Vector;import gnu.java.io.ObjectIdentityWrapper;import gnu.java.lang.reflect.TypeSignature;import java.lang.reflect.Field;import java.lang.reflect.Method;import java.lang.reflect.InvocationTargetException;import gnu.classpath.Configuration;public class ObjectInputStream extends InputStream implements ObjectInput, ObjectStreamConstants{ /** Creates a new <code>ObjectInputStream</code> that will do all of its reading from <code>in</code>. This method also checks the stream by reading the header information (stream magic number and stream version). @exception IOException Reading stream header from underlying stream cannot be completed. @exception StreamCorruptedException An invalid stream magic number or stream version was read from the stream. @see readStreamHeader () */ public ObjectInputStream (InputStream in) throws IOException, StreamCorruptedException { if (Configuration.DEBUG) { String val = System.getProperty("gcj.dumpobjects"); if (dump == false && val != null && !val.equals("")) { dump = true; System.out.println ("Serialization debugging enabled"); } else if (dump == true && (val == null || val.equals(""))) { dump = false; System.out.println ("Serialization debugging disabled"); } } this.resolveEnabled = false; this.isDeserializing = false; this.blockDataPosition = 0; this.blockDataBytes = 0; this.blockData = new byte[BUFFER_SIZE]; this.blockDataInput = new DataInputStream (this); this.realInputStream = new DataInputStream (in); this.nextOID = baseWireHandle; this.objectLookupTable = new Hashtable (); this.validators = new Vector (); setBlockDataMode (true); readStreamHeader (); } /** Returns the next deserialized object read from the underlying stream. This method can be overriden by a class by implementing <code>private void readObject (ObjectInputStream)</code>. If an exception is thrown from this method, the stream is left in an undefined state. @exception ClassNotFoundException The class that an object being read in belongs to cannot be found. @exception IOException Exception from underlying <code>InputStream</code>. */ public final Object readObject () throws ClassNotFoundException, IOException { if (this.useSubclassMethod) return readObjectOverride (); boolean was_deserializing; Object ret_val; was_deserializing = this.isDeserializing; boolean is_consumed = false; boolean old_mode = setBlockDataMode (false); this.isDeserializing = true; byte marker = this.realInputStream.readByte (); dumpElement ("MARKER: 0x" + Integer.toHexString(marker) + " "); try { switch (marker) { case TC_ENDBLOCKDATA: { ret_val = null; is_consumed = true; break; } case TC_BLOCKDATA: case TC_BLOCKDATALONG: { if (marker == TC_BLOCKDATALONG) dumpElementln ("BLOCKDATALONG"); else dumpElementln ("BLOCKDATA"); readNextBlock (marker); throw new StreamCorruptedException ("Unexpected blockData"); } case TC_NULL: { dumpElementln ("NULL"); ret_val = null; break; } case TC_REFERENCE: { dumpElement ("REFERENCE "); Integer oid = new Integer (this.realInputStream.readInt ()); dumpElementln (Integer.toHexString(oid.intValue())); ret_val = ((ObjectIdentityWrapper) this.objectLookupTable.get (oid)).object; break; } case TC_CLASS: { dumpElementln ("CLASS"); ObjectStreamClass osc = (ObjectStreamClass)readObject (); Class clazz = osc.forClass (); assignNewHandle (clazz); ret_val = clazz; break; } case TC_PROXYCLASSDESC: { dumpElementln ("PROXYCLASS"); int n_intf = this.realInputStream.readInt(); String[] intfs = new String[n_intf]; for (int i = 0; i < n_intf; i++) { intfs[i] = this.realInputStream.readUTF(); System.out.println(intfs[i]); } boolean oldmode = setBlockDataMode (true); Class cl = resolveProxyClass(intfs); setBlockDataMode(oldmode); ObjectStreamClass osc = ObjectStreamClass.lookup(cl); assignNewHandle (osc); if (!is_consumed) { byte b = this.realInputStream.readByte (); if (b != TC_ENDBLOCKDATA) throw new IOException ("Data annotated to class was not consumed." + b); } else is_consumed = false; ObjectStreamClass superosc = (ObjectStreamClass)readObject (); osc.setSuperclass (superosc); ret_val = osc; break; } case TC_CLASSDESC: { dumpElement ("CLASSDESC NAME="); String name = this.realInputStream.readUTF (); dumpElement (name + "; UID="); long uid = this.realInputStream.readLong (); dumpElement (Long.toHexString(uid) + "; FLAGS="); byte flags = this.realInputStream.readByte (); dumpElement (Integer.toHexString(flags) + "; FIELD COUNT="); short field_count = this.realInputStream.readShort (); dumpElementln (Short.toString(field_count)); ObjectStreamField[] fields = new ObjectStreamField[field_count]; ObjectStreamClass osc = new ObjectStreamClass (name, uid, flags, fields); assignNewHandle (osc); for (int i=0; i < field_count; i++) { dumpElement (" TYPE CODE="); char type_code = (char)this.realInputStream.readByte (); dumpElement (type_code + "; FIELD NAME="); String field_name = this.realInputStream.readUTF (); dumpElementln (field_name); String class_name; if (type_code == 'L' || type_code == '[') class_name = (String)readObject (); else class_name = String.valueOf (type_code); // There're many cases you can't get java.lang.Class from // typename if your context class loader can't load it, // then use typename to construct the field fields[i] = new ObjectStreamField (field_name, class_name); } boolean oldmode = setBlockDataMode (true); osc.setClass (resolveClass (osc)); setBlockDataMode (oldmode); if (!is_consumed) { byte b = this.realInputStream.readByte (); if (b != TC_ENDBLOCKDATA) throw new IOException ("Data annotated to class was not consumed." + b); } else is_consumed = false; osc.setSuperclass ((ObjectStreamClass)readObject ()); ret_val = osc; break; } case TC_STRING: case TC_LONGSTRING: { dumpElement ("STRING="); String s = this.realInputStream.readUTF (); dumpElementln (s); ret_val = processResolution (s, assignNewHandle (s)); break; } case TC_ARRAY: { dumpElementln ("ARRAY"); ObjectStreamClass osc = (ObjectStreamClass)readObject (); Class componentType = osc.forClass ().getComponentType (); dumpElement ("ARRAY LENGTH="); int length = this.realInputStream.readInt (); dumpElementln (length + "; COMPONENT TYPE=" + componentType); Object array = Array.newInstance (componentType, length); int handle = assignNewHandle (array); readArrayElements (array, componentType); for (int i=0, len=Array.getLength(array); i < len; i++) dumpElementln (" ELEMENT[" + i + "]=" + Array.get(array, i)); ret_val = processResolution (array, handle); break; } case TC_OBJECT: { dumpElementln ("OBJECT"); ObjectStreamClass osc = (ObjectStreamClass)readObject (); Class clazz = osc.forClass (); if (!Serializable.class.isAssignableFrom (clazz)) throw new NotSerializableException (clazz + " is not Serializable, and thus cannot be deserialized."); if (Externalizable.class.isAssignableFrom (clazz)) { Externalizable obj = null; try { obj = (Externalizable)clazz.newInstance (); } catch (InstantiationException e) { throw new ClassNotFoundException ("Instance of " + clazz + " could not be created"); } catch (IllegalAccessException e) { throw new ClassNotFoundException ("Instance of " + clazz + " could not be created because class or zero-argument constructor is not accessible"); } catch (NoSuchMethodError e) { throw new ClassNotFoundException ("Instance of " + clazz + " could not be created because zero-argument constructor is not defined"); } int handle = assignNewHandle (obj); boolean read_from_blocks = ((osc.getFlags () & SC_BLOCK_DATA) != 0); boolean oldmode = this.readDataFromBlock; if (read_from_blocks) setBlockDataMode (true); obj.readExternal (this); if (read_from_blocks) setBlockDataMode (oldmode); ret_val = processResolution (obj, handle); break; } // end if (Externalizable.class.isAssignableFrom (clazz)) // find the first non-serializable, non-abstract // class in clazz's inheritance hierarchy Class first_nonserial = clazz.getSuperclass (); while (Serializable.class.isAssignableFrom (first_nonserial) || Modifier.isAbstract (first_nonserial.getModifiers ())) first_nonserial = first_nonserial.getSuperclass (); Object obj = null; obj = newObject (clazz, first_nonserial); if (obj == null) throw new ClassNotFoundException ("Instance of " + clazz + " could not be created"); int handle = assignNewHandle (obj); this.currentObject = obj; ObjectStreamClass[] hierarchy = ObjectStreamClass.getObjectStreamClasses (clazz); for (int i=0; i < hierarchy.length; i++) { this.currentObjectStreamClass = hierarchy[i]; dumpElementln ("Reading fields of " + this.currentObjectStreamClass.getName ()); // XXX: should initialize fields in classes in the hierarchy // that aren't in the stream // should skip over classes in the stream that aren't in the // real classes hierarchy if (this.currentObjectStreamClass.hasReadMethod()) { fieldsAlreadyRead = false; boolean oldmode = setBlockDataMode (true); callReadMethod (obj, this.currentObjectStreamClass); setBlockDataMode (oldmode); dumpElement ("ENDBLOCKDATA? "); try { // FIXME: XXX: This try block is to catch EOF which is // thrown for some objects. That indicates a bug in the logic. if (this.realInputStream.readByte () != TC_ENDBLOCKDATA) throw new IOException ("No end of block data seen for class with readObject (ObjectInputStream) method."); dumpElementln ("yes"); } catch (EOFException e) { dumpElementln ("no, got EOFException"); } catch (IOException e) { dumpElementln ("no, got IOException"); } } else { readFields (obj, currentObjectStreamClass); } } this.currentObject = null; this.currentObjectStreamClass = null; ret_val = processResolution (obj, handle); break;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -