`
javantsky
  • 浏览: 84259 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

ArrayList序列化的一点理解

    博客分类:
  • java
阅读更多

今天同事问到ArrayList中的

private transient E[] elementData;

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

我的回答是ArrayList重写了

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了下,发现主流说法如下:

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,如下:

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序列化功能:

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);
		}

 输出结果为:

sssssssssssssssss
bbbbbbbbbbbbbbbbbbt
gggggggggggggggggg

 

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

 

那是为什么呢?

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

                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);
		}

 输出结果为:

mmmmmmmmmm
null
nnnnnnnnnn
null

 

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

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

ArrayList是会开辟多余空间来保存数据的,而系列化和反序列化这些没有存放数据的空间是要消耗更多资源的,所以ArrayList的数组就声明为transient,告诉虚拟机这个你别管,我自己来处理,然后就自己实现write/readObject方法,仅仅系列化已经存放的数据。

 

 

 

14
10
分享到:
评论
8 楼 enjoyj2ee 2011-01-13  
最近  要保存一个 很复杂的构建的对象 ,没有用java原生的序列化玩法,直接
用 XStream 序列对象 成xml ,感觉 很爽,对象还是可视的。
也不用考虑实现 serialible..
7 楼 javantsky 2011-01-13  
kakaluyi 写道
贴出readObject方法就知道了,确实和楼主说的一样,这样可以少序列化一些null对象,节约一些序列化后的存储空间。
private void readObject(java.io.ObjectInputStream s)
        throws java.io.IOException, ClassNotFoundException {
// Read in size, and any hidden stuff
s.defaultReadObject();

        // Read in array length and allocate array
        int arrayLength = s.readInt();
        Object[] a = elementData = new Object[arrayLength];

// Read in all elements in the proper order.
for (int i=0; i<size; i++)
            a[i] = s.readObject();
    }
楼主继续努力,希望能够有更多底层的代码可以分享


多谢兄台的补充,需要说明下,一般来说如果自己实现了writeObject方法,是需要自己实现readObject方法的。
6 楼 kakaluyi 2011-01-13  
贴出readObject方法就知道了,确实和楼主说的一样,这样可以少序列化一些null对象,节约一些序列化后的存储空间。
private void readObject(java.io.ObjectInputStream s)
        throws java.io.IOException, ClassNotFoundException {
// Read in size, and any hidden stuff
s.defaultReadObject();

        // Read in array length and allocate array
        int arrayLength = s.readInt();
        Object[] a = elementData = new Object[arrayLength];

// Read in all elements in the proper order.
for (int i=0; i<size; i++)
            a[i] = s.readObject();
    }
楼主继续努力,希望能够有更多底层的代码可以分享
5 楼 t42dw 2011-01-13  
受教了.....
4 楼 juda 2011-01-12  
了然,非常感谢,哈哈
javantsky 写道
juda 写道
"ArrayList是会开辟多余空间来保存数据的,而系列化和反序列化这些没有存放数据的空间是要消耗更多资源的,所以ArrayList的数组就声明为transient,自己实现write/readObject方法,仅仅系列化已经存放的数据"解释的不是很清楚。请详细说下


在new一个ArrayList对象时,默认是开辟一个长度为10的对象数组,如果只存入3个对象,如果采用默认序列化,则会将其余7个null也序列化到文件中。

如果声明为transient类型,则告诉虚拟机,这个你别帮我弄了,我自己来处理,这才有了ArrayList的write/readObject方法。

不知兄台对这种解释是否满意呢?
3 楼 javantsky 2011-01-12  
juda 写道
"ArrayList是会开辟多余空间来保存数据的,而系列化和反序列化这些没有存放数据的空间是要消耗更多资源的,所以ArrayList的数组就声明为transient,自己实现write/readObject方法,仅仅系列化已经存放的数据"解释的不是很清楚。请详细说下


在new一个ArrayList对象时,默认是开辟一个长度为10的对象数组,如果只存入3个对象,如果采用默认序列化,则会将其余7个null也序列化到文件中。

如果声明为transient类型,则告诉虚拟机,这个你别帮我弄了,我自己来处理,这才有了ArrayList的write/readObject方法。

不知兄台对这种解释是否满意呢?
2 楼 liuleigang 2011-01-12  
这个解释确实靠谱。仅序列化已有的数据比较经济。
1 楼 juda 2011-01-12  
"ArrayList是会开辟多余空间来保存数据的,而系列化和反序列化这些没有存放数据的空间是要消耗更多资源的,所以ArrayList的数组就声明为transient,自己实现write/readObject方法,仅仅系列化已经存放的数据"解释的不是很清楚。请详细说下

相关推荐

    Java对象的序列化和反序列化实践

    在Java编程语言中,对象的序列化和反序列化是两个关键的概念,它们允许我们将对象的状态转换为可存储或传输的格式,然后再恢复为原始对象。这个过程对于数据持久化、网络传输以及跨进程通信等场景非常有用。下面将...

    C#序列化和反序列化

    ### C#中的序列化与反序列化 #### 一、概念理解 序列化与反序列化是编程领域中经常遇到的概念。简而言之,**序列化**是指将对象的状态(即对象的数据)转换为可以存储或传输的形式的过程;而**反序列化**则是将...

    序列化(三) 实例分析深入了解序列化

    `java.util.Optional`、`java.util.ArrayList`、`java.util.HashMap`等内置类都实现了序列化,方便我们在实际开发中使用。 10. **源码分析**: 博文中可能深入分析了`ObjectOutputStream`和`ObjectInputStream`的...

    ArrayList源码分析(含jdk1.8).pdf

    为了更好地理解ArrayList的工作原理,我们还需关注其序列化和反序列化过程。由于ArrayList内部数组elementData被标记为transient,意味着它不会被默认的序列化机制处理。在ArrayList的序列化过程中,为了保持序列化...

    Android Parcelable序列化自定义类集合在Activity间传递

    综上所述,要理解并实现Android中Parcelable序列化自定义类集合在Activity间传递,开发者需要掌握Parcelable接口的使用、自定义类的Parcelable实现、集合处理、Intent数据传递,以及可能借助的源码阅读和辅助工具。...

    序列化与集合窗体应用

    在“序列化与集合窗体应用”项目中,可能涵盖了如何将集合对象(如ArrayList或Dictionary)进行序列化,然后保存到文件或数据库中。这在处理大量数据时非常有用,因为可以直接读取序列化的数据,避免了重新计算或...

    c# arraylist

    本篇文章详细介绍了如何在C#中使用`ArrayList`与数据库进行交互,包括序列化和反序列化的过程、参数化查询的实现以及内存流的使用。这些知识点不仅适用于本案例,也是开发过程中非常实用的技术。理解这些概念有助于...

    intent传递ArrayList&lt;T&gt;

    对于非基本类型,如自定义对象,需要该对象实现Parcelable接口,因为Parcelable接口提供了序列化和反序列化的方法,使对象能够被Intent安全地传递。 1. 实现Parcelable接口:为了将ArrayList中的每个元素(T)传递...

    ArrayList的底层原理Java系列2021.pdf

    由于ArrayList可能包含大量的数据,为了避免不必要的序列化操作,ArrayList中的elementData数组被标记为transient,以表明在序列化过程中不应被默认序列化机制处理。 在进行序列化时,可以使用transient关键字标记...

    arrayList原理说明

    该数组是`transient`类型的,这意味着当序列化`ArrayList`对象时,不会保存该数组的信息,只保存列表元素本身及其相关的状态信息。 #### 构造方法 `ArrayList`提供了多种构造方法来适应不同的应用场景: 1. **无参...

    第二章 ArrayList源码解析1

    ArrayList通过自定义的`writeObject`和`readObject`方法来进行序列化和反序列化,而不是依赖于默认机制。 在`writeObject`方法中,首先序列化元素数量,然后逐个序列化数组中的所有元素。而在`readObject`方法中,...

    Java员工管理系统,ArrayList存储

    可以考虑使用文件存储(序列化与反序列化)、数据库存储(如JDBC连接MySQL)等方式。 6. **界面交互**:为了用户友好,可以设计一个简单的命令行界面或者图形用户界面(GUI),让用户能够方便地输入命令进行操作。...

    java集合,多线程,序列化等基础练习源码

    Java集合、多线程以及序列化是Java编程中不可或缺的基础概念。这些技术广泛应用于各种复杂的系统设计和开发中,理解并熟练掌握它们对于提升Java程序员的技能至关重要。 首先,我们来详细探讨Java集合。Java集合框架...

    ArrayList底层原理

    Serializable接口使得ArrayList的实例可以被序列化和反序列化,这意味着ArrayList对象可以被转换为字节流并保存到文件或网络上,之后再重新加载回内存。而Cloneable接口则是为了支持对象的浅复制,即通过调用Object...

    ArrayList源码解析(数据结构及底层实现)(csdn)————程序.pdf

    ArrayList 在序列化时仅保存实际存储的元素数量和元素,反序列化时根据这些信息重建数组,确保只有实际存在的元素被还原。 ### 2. 构造方法 ArrayList 提供了几个构造方法,其中最常见的是无参构造方法,它会初始...

    利用Handler+ArrayList来处理并发事件

    描述中的“SerialEventDemo”可能是一个演示程序,展示了如何序列化处理并发事件。ArrayList在此场景下可能是用来存储待处理事件的容器。ArrayList是Java中的动态数组,可以方便地添加、删除和访问元素,但在并发...

    46-Java知识点 手写ArrayList1

    如果我们想要序列化 ArrayList,需要使用 writeObject 和 readObject 方法。 扩容 ArrayList 的扩容是非常重要的步骤。在添加元素时,如果数组的容量不足,需要扩容数组。扩容的步骤如下: 1. 传入一个要扩容的值...

    华为OD机试C卷- 模拟数据序列化传输(Java & JS & Python & C & C++).md-私信看全套

    本题目旨在考察编程者对于序列化与反序列化的理解和实现能力。序列化通常指的是将数据结构或对象状态转换为可以存储或传输的形式的过程;而反序列化则是指将序列化后的数据恢复到其原始形式。这些技术广泛应用于网络...

Global site tag (gtag.js) - Google Analytics