A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

看了半天没明白序列化是干嘛的,怎么应用

5 个回复

正序浏览
1、序列化是干什么的?    简单说就是为了保存在内存中的各种对象的状态,并且可以把保存的对象状态再读出来。虽然你可以用你自己的各种各样的方法来保存Object States,但是Java给你提供一种应该比你自己好的保存对象状态的机制,那就是序列化。  2、什么情况下需要序列化   a)当你想把的内存中的对象保存到一个文件中或者数据库中时候; b)当你想用套接字在网络上传送对象的时候; c)当你想通过RMI传输对象的时候;  3、当对一个对象实现序列化时,究竟发生了什么?  在没有序列化前,每个保存在堆(Heap)中的对象都有相应的状态(state),即实例变量(instance ariable)比如:  Foo myFoo = new Foo();  myFoo .setWidth(37);  myFoo.setHeight(70);     当通过下面的代码序列化之后,MyFoo对象中的width和Height实例变量的值(37,70)都被保存到foo.ser文件中,这样以后又可以把它 从文件中读出来,重新在堆中创建原来的对象。当然保存时候不仅仅是保存对象的实例变量的值,JVM还要保存一些小量信息,比如类的类型等以便恢复原来的对 象。  FileOutputStream fs = new FileOutputStream("foo.ser");  ObjectOutputStream os = new ObjectOutputStream(fs);  os.writeObject(myFoo);   4、实现序列化(保存到一个文件)的步骤  a)Make a FileOutputStream  java 代码 FileOutputStream fs = new FileOutputStream("foo.ser");  b)Make a ObjectOutputStream   java 代码 ObjectOutputStream os = new ObjectOutputStream(fs);  c)write the object  java 代码 os.writeObject(myObject1);  os.writeObject(myObject2);  os.writeObject(myObject3);  d) close the ObjectOutputStream  java 代码 os.close();   5、举例说明  java 代码 import java.io.*;   public class Box implements Serializable  {  private int width;  private int height;   public void setWidth(int width){  this.width = width;  }  public void setHeight(int height){  this.height = height;  }   public static void main(String[] args){  Box myBox = new Box();  myBox.setWidth(50);  myBox.setHeight(30);   try{  FileOutputStream fs = new FileOutputStream("foo.ser");  ObjectOutputStream os = new ObjectOutputStream(fs);  os.writeObject(myBox);  os.close();  }catch(Exception ex){  ex.printStackTrace();  }  }   }   6、相关注意事项  a)当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口; b)当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化; c)并非所有的对象都可以序列化,,至于为什么不可以,有很多原因了,比如:    1.安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行rmi传输 等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的。   2. 资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分 配,而且,也是没有必要这样实现。
回复 使用道具 举报
对象状态的持久化
回复 使用道具 举报
这样说你就懂了。任何类型都是数据,自定义对象也是数据类型,数据类型是由字节组成,字节是由位组成,计算机操作的都是位,你把一个字符打印到文件中其实打印的是这个字符的编号集对应的编号转换成的位,也就是2进制。任何数据的存与取都必须先转换成2进制,再由另一端的设备转换回来。对象要存取也需要这么干。将对象序列化后其实就是将对象在内存中的二进制表现形式打印到相关的流设备中。等到另一端设备收到数据后再使用有规律的方式重新重组合 ,这个规律就是对象,或是说Serializabl提供的唯一区别码也可以。

换个说法,假如你现在有1吨大米,你想把他们运走,可是你只有自行车,那么你要做的就是把1吨大米分成斤,然后一斤一斤往外运,到了目的地再重新组合成一吨大米。序列化也是一样。你只有流,流只能放二进制数据,那么你就只有把对象分成二进制传送过去,然后到了目的地再重组。

这个问题主要是因为Java语言的无指针性产生的,在C|C++中。直接将一个对象的头地址,加这个对象的字节长度封装成一个数据包扔到Socket连接或文件中就完事了。Java没有指针,也就没有地址,更不可能有内存长度,所以只有通过ObjectOutputStream将外传送一段字节码。而这个字节码取多长取决于这个对象占用的内存位宽有多大。
回复 使用道具 举报
贾永长 发表于 2013-6-30 21:57
这个挺详细的!http://zhidao.baidu.com/question/277515269.html

谢谢!还是不怎么懂,看来得慢慢渗透
回复 使用道具 举报
这个挺详细的!{:soso_e130:}http://zhidao.baidu.com/question/277515269.html
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马