《Java安全编码标准》一1.6 序列化

本文介绍了Java中的序列化机制,解释了如何将对象的状态保存到字节流中以便于将来恢复。文章详细阐述了序列化的过程,包括对象的序列化与反序列化操作,并讨论了在序列化过程中对对象字段的影响及安全考虑。

1.6 序列化

序列化使得Java程序中对象的状态能被抓取并写入到字节流中[Sun 2004b]。这使得该对象的状态能够保存下来,所以可以在未来需要的时候恢复(通过反序列化)。序列化可以通过网络使用RMI的方式调用Java方法,这些对象被编组(序列化),在分布部署的虚拟机之间进行交换,然后进行解组(反序列化)。序列化还广泛使用在Java Beans中。
可以用以下方式序列化对象:

ObjectOutputStream oos = new ObjectOutputStream(
  new FileOutputStream("SerialOutput"));
oos.writeObject(someObject);
oos.flush();

可以用以下的方式反序列化对象:

ObjectInputStream ois = new ObjectInputStream(
  new FileInputStream("SerialOutput"));
someObject = (SomeClass) ois.readObject();

序列化能够捕捉对象中的所有非暂态字段,包括那些正常情况下不能读取的非公有字段,这通过object类的Serializable接口来实现。序列化的值写入字节流之后,如果字节流是可读的,那么原先那些正常情况下不能访问的字段的值也就可以通过推导得出。此外访问,当反序列化一个类时,原来保存的值可能会被修改或者伪造。
引进一个安全管理器并不能防止正常情况下无法访问的字段被序列化和反序列化(如果字节流需要被存储或发送,那么必须给予它从文件或网络进行读写的权限)。网络流量(包括RMI)可以得到保护,但是,这需要使用SSL/TLS协议。
在序列化和反序列化对象时需要进行特殊处理的类可以实现有以下签名的方法[API 2006]:

private void writeObject(java.io.ObjectOutputStream out)
 throws IOException;
private void readObject(java.io.ObjectInputStream in)
 throws IOException, ClassNotFoundException;

如果一个Serializable类没有重写writeObject()方法的实现,这个对象会使用默认的方法来完成序列化,它会序列化所有的public、protected、package-private,以及private字段,除了transient字段。同样,当Serializable类没有重写readObject()方法的实现时,这个对象会反序列化所有的public、protected,以及private字段,除了那些transient字段。这个问题会在规则SER01-J中深入探讨。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值