`
kakaluyi
  • 浏览: 437907 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

ArrayList序列化

阅读更多

今天同事问到ArrayList中的

Java代码 复制代码
  1. private transient E[] elementData;  
private transient E[] elementData;

 声明为transient,为什么还可以序列化成功呢?

我的回答是ArrayList重写了

Java代码 复制代码
  1. private void writeObject(java.io.ObjectOutputStream s)   
  2.         throws java.io.IOException{   
  3.     int expectedModCount = modCount;   
  4.     // Write out element count, and any hidden stuff   
  5.     s.defaultWriteObject();   
  6.   
  7.         // Write out array length   
  8.         s.writeInt(elementData.length);   
  9.   
  10.     // Write out all elements in the proper order.   
  11.     for (int i=0; i<size; i++)   
  12.             s.writeObject(elementData[i]);   
  13.   
  14.     if (modCount != expectedModCount) {   
  15.         throw new ConcurrentModificationException();   
  16.     }   
  17.     }  
private void writeObject(java.io.ObjectOutputStream s)
        throws java.io.IOException{
	int expectedModCount = modCount;
	// Write out element count, and any hidden stuff
	s.defaultWriteObject();

        // Write out array length
        s.writeInt(elementData.length);

	// Write out all elements in the proper order.
	for (int i=0; i<size; i++)
            s.writeObject(elementData[i]);

 	if (modCount != expectedModCount) {
	    throw new ConcurrentModificationException();
	}
    }

 在使用ObjectOutputStream序列化对象时会调用这个writeObject方法。

 

第二个问题是为什么要声明为transient呢?

在google了下,发现主流说法如下:

Java代码 复制代码
  1. ArrayList实现了java.io.Serializable接口,所以ArrayList对象可以序列化到持久存储介质中。ArrayList的主要属性定义如下:   
  2.   
  3.     * private static final long serialVersionUID = 8683452581122892189L;   
  4.     * private transient Object elementData[];   
  5.     * private int size;   
  6.   
  7.   
  8. 可以看出serialVersionUID和size都将自动序列化到介质中,但elementData数组对象却定义为transient了。   
  9. 也就是说 ArrayList中的所有这些元素都不会自动系列化到介质中。为什么要这样实现?因为elementData数组中存储的   
  10. “元素”其实仅是对这些元素的一个引用,并不是真正的对象,序列化一个对象的引用是毫无意义的,因为序列化是为了   
  11. 反序列化,当你反序列化时,这些对象的引用已经不可能指向原来的对象了。所以在这儿需要手工的对ArrayList的元素进   
  12. 行序列化操作。这就是writeObject()的作用。  
ArrayList实现了java.io.Serializable接口,所以ArrayList对象可以序列化到持久存储介质中。ArrayList的主要属性定义如下:

    * private static final long serialVersionUID = 8683452581122892189L;
    * private transient Object elementData[];
    * private int size;


可以看出serialVersionUID和size都将自动序列化到介质中,但elementData数组对象却定义为transient了。
也就是说 ArrayList中的所有这些元素都不会自动系列化到介质中。为什么要这样实现?因为elementData数组中存储的
“元素”其实仅是对这些元素的一个引用,并不是真正的对象,序列化一个对象的引用是毫无意义的,因为序列化是为了
反序列化,当你反序列化时,这些对象的引用已经不可能指向原来的对象了。所以在这儿需要手工的对ArrayList的元素进
行序列化操作。这就是writeObject()的作用。

 果真如此么??????

验证下:

把ArrayList的内容完全copy到一个新类里面,命名为MyArrayList,如下:

Java代码 复制代码
  1. public class MyArrayList<E> extends AbstractList<E>   
  2.         implements List<E>, RandomAccess, Cloneable, java.io.Serializable   
  3. {   
  4.     private static final long serialVersionUID = 8683452581122892189L;   
  5.   
  6.     /**  
  7.      * The array buffer into which the elements of the ArrayList are stored.  
  8.      * The capacity of the ArrayList is the length of this array buffer.  
  9.      */  
  10.     private E[] elementData;   
  11.   
  12.    。。。。。。。。   
  13.   
  14.    private void writeObject(java.io.ObjectOutputStream s)   
  15.         throws java.io.IOException{   
  16.     int expectedModCount = modCount;   
  17.     // Write out element count, and any hidden stuff   
  18.     s.defaultWriteObject();   
  19.   
  20.     if (modCount != expectedModCount) {   
  21.         throw new ConcurrentModificationException();   
  22.     }   
  23.     }   
  24.   
  25.     /**  
  26.      * Reconstitute the <tt>ArrayList</tt> instance from a stream (that is,  
  27.      * deserialize it).  
  28.      */  
  29.     private void readObject(java.io.ObjectInputStream s)   
  30.         throws java.io.IOException, ClassNotFoundException {   
  31.     // Read in size, and any hidden stuff   
  32.     s.defaultReadObject();   
  33.      }   
  34.   
  35. }  
public class MyArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
    private static final long serialVersionUID = 8683452581122892189L;

    /**
     * The array buffer into which the elements of the ArrayList are stored.
     * The capacity of the ArrayList is the length of this array buffer.
     */
    private E[] elementData;

   。。。。。。。。

   private void writeObject(java.io.ObjectOutputStream s)
        throws java.io.IOException{
	int expectedModCount = modCount;
	// Write out element count, and any hidden stuff
	s.defaultWriteObject();

 	if (modCount != expectedModCount) {
	    throw new ConcurrentModificationException();
	}
    }

    /**
     * Reconstitute the <tt>ArrayList</tt> instance from a stream (that is,
     * deserialize it).
     */
    private void readObject(java.io.ObjectInputStream s)
        throws java.io.IOException, ClassNotFoundException {
	// Read in size, and any hidden stuff
	s.defaultReadObject();
     }

}

 把transient去掉,write/readObject采用默认方式。

测试下MyArraylist序列化功能:

Java代码 复制代码
  1. MyArrayList al = new MyArrayList<String>();   
  2.         al.add("sssssssssssssssss");   
  3.         al.add("bbbbbbbbbbbbbbbbbbt");   
  4.         al.add("gggggggggggggggggg");   
  5.            
  6.         ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\al.tmp"));   
  7.         oos.writeObject(al);   
  8.            
  9.         ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\al.tmp"));   
  10.            
  11.         MyArrayList<String> a = (MyArrayList<String>)ois.readObject();   
  12.         for(String s: a)   
  13.         {   
  14.             System.out.println(s);   
  15.         }  
MyArrayList al = new MyArrayList<String>();
		al.add("sssssssssssssssss");
		al.add("bbbbbbbbbbbbbbbbbbt");
		al.add("gggggggggggggggggg");
		
		ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\al.tmp"));
		oos.writeObject(al);
		
		ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\al.tmp"));
		
		MyArrayList<String> a = (MyArrayList<String>)ois.readObject();
		for(String s: a)
		{
			System.out.println(s);
		}

 输出结果为:

Java代码 复制代码
  1. sssssssssssssssss   
  2. bbbbbbbbbbbbbbbbbbt   
  3. gggggggggggggggggg  
sssssssssssssssss
bbbbbbbbbbbbbbbbbbt
gggggggggggggggggg

 

到此证明:引用序列化无效的说法是错误的,这点在ObjectOutputStream中也有说明。

 

那是为什么呢?

既然是数组,要序列化到文件中,那就单独测试下数组对象的序列化和反序列化吧

Java代码 复制代码
  1.               String[] stra = new String[4];   
  2. stra[0] = "mmmmmmmmmm";   
  3. stra[2] = "nnnnnnnnnn";   
  4.   
  5. oos = new ObjectOutputStream(new FileOutputStream("D:\\sa.tmp"));   
  6. oos.writeObject(stra);   
  7.   
  8. ois = new ObjectInputStream(new FileInputStream("D:\\sa.tmp"));   
  9.   
  10. String[] str  = (String[])ois.readObject();   
  11. for(String s: str)   
  12. {   
  13.     System.out.println(s);   
  14. }  
                String[] stra = new String[4];
		stra[0] = "mmmmmmmmmm";
		stra[2] = "nnnnnnnnnn";
		
		oos = new ObjectOutputStream(new FileOutputStream("D:\\sa.tmp"));
		oos.writeObject(stra);
		
		ois = new ObjectInputStream(new FileInputStream("D:\\sa.tmp"));
		
		String[] str  = (String[])ois.readObject();
		for(String s: str)
		{
			System.out.println(s);
		}

 输出结果为:

Java代码 复制代码
  1. mmmmmmmmmm   
  2. null  
  3. nnnnnnnnnn   
  4. null  
mmmmmmmmmm
null
nnnnnnnnnn
null

 

从输出结果来看,数组序列化时,不管是否有值,都会将整个数组序列化到文件中。

由此可以看出,比较靠谱的原因是:

ArrayList是会开辟多余空间来保存数据的,而系列化和反序列化这些没有存放数据的空间是要消耗更多资源的,所以ArrayList的数组就声明为transient,自己实现write/readObject方法,仅仅系列化已经存放的数据。

分享到:
评论
3 楼 绝无仅有的霸气 2014-09-17  
“...因为序列化是为了反序列化,当你反序列化时,这些对象的引用已经不可能指向原来的对象了”,你在本地的一个程序中实现序列化和反序列化,这些引用应该还在吧。
2 楼 腹黑君 2013-10-28  
我看他源码的时候,就在疑惑他这一点。。。
1 楼 zhangyeny 2011-08-26  
楼主真的很强悍

相关推荐

    浅谈C# 序列化与反序列化几种格式的转换

    下面小编就为大家带来一篇浅谈C# 序列化与反序列化几种格式的转换。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧

    深入分析Java的序列化与反序列化

    本文通过分析ArrayList的序列化来介绍Java序列化的相关内容。主要涉及到以下几个问题:  怎么实现Java的序列化  为什么实现了java.io.Serializable接口才能被序列化  transient的作用是什么  怎么自定义序列...

    C# 序列化与反序列化

    C# XmlSerializer 分别为对ArrayList的以及对类的实例化对象的序列化与反序列化

    序列化Hashtable

     那先来看一个可以正常序列化的集合类――Arraylist:serialize = new XmlSerializer(typeof(ArrayList));在序列化过程中临时生成的文件中可以发现如下的语句: WriteStartElement(@&quot;ArrayOfAnyType&quot;, @&...

    Runtime-Serialization:在运行时高效地序列化数据和场景!

    运行时序列化 在运行时高效地序列化数据和场景! 运行时序列化提供了强大,... •提供序列化回调笔记: •Metro / WP8平台不支持Hashtable和Arraylist类,因此我们不支持上述类型的序列化。 随时使用分享您的反馈意见

    Android中的序列化浅析

    序列化原因 序列化的原因基本可以归纳为以下三种情况: 1.永久性保存对象,保存对象的字节序列到本地文件中; 2.对象在网络中传递; 3.对象在IPC间传递。 序列化方法 在Android系统中关于序列化的方法一般有两种,...

    Java序列化类库protostuff.zip

    protostuff 是一个支持各种格式的一个序列化Java类库,包括 JSON、XML、YAML等格式。示例代码:public class UserSchema implements Schema {  public boolean isInitialized(User user)  {  return...

    S22-XMLDecoder反序列化1

    &lt;object class="java.util.ArrayList"&gt;一开始想直接序列化一个 Runtime 对象,后来发现这个对象不能被 Encoder ,

    理解Java的序列化与反序列化

    主要为大家详细介绍了Java的序列化与反序列化,序列化是一种对象持久化的手段。普遍应用在网络传输、RMI等场景中。本文通过分析ArrayList的序列化来介绍Java序列化的相关内容,感兴趣的小伙伴们可以参考一下

    里氏替换原则+排序_接口_泛型比较器+序列化+泛型集合.rar

    简单介绍 里氏替换原则 简单工程模式 实现任意排序 使用了 is 和 as 方法 ArrayList数组 与 List泛型集合 Hashtable数组 与 Dictionary,v&gt;泛型集合 的基本使用 序列化 将数据存储

    python-javaobj:python-javaobj是一个python库,提供了读取Java对象序列化ObjectOutputStream的函数

    python-javaobj 是一个 python 库,它提供了读取和写入(写入当前是 WIP)Java 对象的函数,序列化或将被ObjectOutputStream反序列化。 这种对象表示形式是 Java 世界中的标准数据交换格式。 javaobj 模块公开了...

    java面向对象课程设计——学生管理系统——附源码+word文档

    (1)学生数据按对象序列化写入student.dat文件 (2)程序启动后,从student.dat文件读取学生对象存入ArrayList中; (3)增加学生数据时将学生对象Student存入ArrayList,按保存后再写入student.dat; (4)删除学生必须...

    Gson_and_HttpComponent:测试Gson和HttpComponent库,序列化arrayList并使用REST服务

    Gson和HttpComponent 测试Java Gson和HttpComponent库,序列化arrayList并使用REST服务

    List集合之ArrayList

    ArrayList集成AbstractList抽象类,实现了List、RandomAccess、Cloneable、java.io.Serializable这四个接口,其中我们可以看到,实现了Cloneable和Serializable接口就代表着ArrayList是支持克隆和序列化的,这里有个...

    Android编程实现使用Intent传输包含自定义类的ArrayList示例

    Java的对象序列化指的是将那些实现了Serializable接口的对象转换成一个字节序列,并且能在需要的时候再将这个字节序列完全恢复为之前的对象。 想实现对象的序列化,需要实现java.io.Serializable接口(注意,这个接口...

    gson 的使用

    gson 的常用语法使用, 包括: 1. 将ArrayList字符串转换为 JSON 2. 将JSON字符串转换为 ArrayList 3. 将HashMap字符串转换为 JSON 4. 将JSON字符串转换为 HashMap 5. 反序列化方法

    ArrayList 源码深度解析

    ArrayList是基于数组实现的List类,封装了一个动态再分配的Object数组,以达到可以动态增长和缩减的索引序列。 长啥样? 如图,是一个长度为6,存储了6个6的数组,下标(index)从0开始,数组(elementData)表示...

    Java实现基于内存的搜索引擎设计和实现.zip

    在序列化过程中充分利用了成员实现的序列化反序列化方法,与在toString方法中调用成员的toString方法有异曲同工之妙 2.充分利用Java实现的API接口。比如在类Index中的getDictionary方法中,需要返回所有保存的Term,...

    DSON-Java:总督序列化对象表示法(http

    Doge序列化对象表示法( )的完整,零依赖Java实现。 支持序列化和解析。 正确地将所有数字从和转换为八进制。 一些代码使用名称,以使Shiba Inu狗更容易理解该代码。 用法 public static class DogeExample { ...

Global site tag (gtag.js) - Google Analytics