摘要: 摘要: 原创出处: http://www.cnblogs.com/Alandre/ 泥沙砖瓦浆木匠 希望转载,保留摘要,谢谢!

钢笔不限贵便宜,书法是来自心对手的交流。-泥沙砖瓦浆木匠

一.对象序列化

当需要存储相同类型的数据,选择固定的长度记录是好选择。但是在面向对象(OOP)程序中,对象之间很少有全部相同的类型。所以,java语言支持一种称为对象序列化(object serialization)的机制。

下面展示一个序列化例子,关于两个对象 Employee 和 Manager 。首先我们必须先把他们存储进去:

ObjectOutputStream out = new ObjectOutputStream( new FileOutputStream("employee.bat"));
Employee employee = new Employee("Employee one ",5000,1989,10,1);
Manager boss = new Manager("Manager one ",8000,1987,7,1);
out.writeObject(employee);
out.writeObject(boss);

为了将这些对象读回,首先要获取一个ObjectInputStream对象,然后用readObject方法获得它们。

ObjectInputStream in = new ObjectInputStream(new FileOutputStream("employee.bat"));
Employee e = (Employee)in.readObject();
Manager b = (Manager)in.readObject();

总而言之,这些希望在对象流存储或者恢复的所有类,必训实现public interface Serializable


类通过实现 java.io.Serializable 接口以启用其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。可序列化类的所有子类型本身都是可序列化的。序列化接口没有方法或字段,仅用于标识可序列化的语义。

其他除了writeObject方法,常用操作的api:

java.io.ObjectOutputStream

writeDouble(double val)
          写入一个 64 位的 double 值。
writeFields()
          将已缓冲的字段写入流中。
writeFloat(float val)
          写入一个 32 位的 float 值。
writeInt(int val)
          写入一个 32 位的 int 值。
writeLong(long val)
          写入一个 64 位的 long 值。
writeObject(Object obj)
          将指定的对象写入 ObjectOutputStream。

二.对象序列化中的算法

就像上面说的,两个Manager 的员工可能是同一个Employee。伪代码如下:

Employee employee = new Employee("Employee one ");
Manager a = new Manager("Manager a");
a.setWorker(employee);
Manager b = new Manager("Manager b");
a.setWorker(employee);

对保存这样的对象,没使用序列化是一种挑战。当类被重新加载时,他可能占据的是原来完全不同的内存地址。对象序列化的原因是:每个对象都用一个序列号(serial number)保存的。如图(两个管理拥有同一个工作人员):

Java的序列化算法序列化算法一般会按步骤做如下事情:

◆将对象实例相关的类元数据输出。

◆递归地输出类的超类描述直到不再有超类。

◆类元数据完了以后,开始从最顶层的超类开始输出对象实例的实际数据值。

◆从上至下递归输出实例的数据

上图(两个管理拥有同一个工作人员)展示是内存中的。一个对象序列化的实例,在磁盘文件中其实是如下图所示。
                                                        

注意:我们这序列化对对象集合保存到磁盘文件。但是序列化另一种非常重要的应用是通过网络将对象集合传送到另一个计算机。序列化用序列号代替了内存地址,他允许将对象集合传到另一台机子。

三.序列化案例

下面,我们就依照上面的Employee 和 Manager对象网络的代码。如下面

ObjectStreamTest.java

package sedion.jeffli.os;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream; public class ObjectStreamTest
{
public static void main(String[] args)
{
Employee employee = new Employee("Employee one", 5000, 1993, 5, 2);
Manage a = new Manage("Manage a", 8000, 1990, 2, 2);
a.setWorker(employee);
Manage b = new Manage("Manage b", 8000, 1990, 2, 2);
b.setWorker(employee); Employee[] staff = new Employee[3]; staff[0] = employee;
staff[1] = a;
staff[2] = b; try
{
ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream("employee.bat"));
outputStream.writeObject(staff);
outputStream.close(); ObjectInputStream inputStream = new ObjectInputStream(new FileInputStream("employee.bat"));
Employee[] newStaff = (Employee[]) inputStream.readObject();
inputStream.close(); newStaff[1].raiseSalary(10); for (Employee e : newStaff)
System.out.println("Employee:"+e);
} catch (Exception e)
{
e.printStackTrace();
}
}
}

另外下面是两个实体类:Employee.java 和 Manage .java

package sedion.jeffli.os;

import java.io.Serializable;
import java.util.Date;
import java.util.GregorianCalendar; public class Employee implements Serializable
{
private static final long serialVersionUID = 1L;
public Employee()
{
} public Employee(String n , double s , int year , int month , int day)
{
name = n;
salary = s;
GregorianCalendar calendar = new GregorianCalendar(year, month - 1, day);
hireDay = calendar.getTime();
} public String getName()
{
return name;
} public double getSalary()
{
return salary;
} public Date getHireDay()
{
return hireDay;
} public void raiseSalary(double byPercent)
{
double raise = salary * byPercent/100;
salary += raise;
} @Override
public String toString()
{
return "Employee [name=" + name + ", salary=" + salary + ", hireDay="
+ hireDay + "]";
} private String name;
private double salary;
private Date hireDay;
}
package sedion.jeffli.os;

public class Manage extends Employee
{
private static final long serialVersionUID = 1L; public Manage(String n , double s , int year , int month , int day)
{
super(n, s, year, month, day);
worker = null;
} public void setWorker(Employee e)
{
worker = e;
} @Override
public String toString()
{
return "Manage [worker=" + worker + "]";
} private Employee worker;
}

我们运行下, ObjectStreamTest.java 会有下面的输出:

Employee:Employee [name=Employee one, salary=5000.0, hireDay=Sun May  :: CST ]
Employee:Manage [worker=Employee [name=Employee one, salary=5000.0, hireDay=Sun May :: CST ]]
Employee:Manage [worker=Employee [name=Employee one, salary=5000.0, hireDay=Sun May :: CST ]]

java.io.ObjectOutputStream

ObjectOutputStream(OutputStream out)
          创建写入指定 OutputStream 的 ObjectOutputStream。

writeObject(Object obj)
          将指定的对象写入 ObjectOutputStream。

java.io.ObjectInputStream

ObjectInputStream(InputStream in)
          创建从指定 InputStream 读取的 ObjectInputStream。

readObject()
          从 ObjectInputStream 读取对象。

四.感谢知识来源和小结

一.对象序列化

二.对象序列化中的算法

三.序列化案例

来自:java基础书籍。

如以上文章或链接对你有帮助的话,别忘了在文章按钮或到页面右下角点击 “赞一个” 按钮哦。你也可以点击页面右边“分享”悬浮按钮哦,让更多的人阅读这篇文章。

动车上的书摘-java对象流与序列化的更多相关文章

  1. Java对象流与序列化学习

    对象流与序列化 对象流有两个类 ObjectOutputStream:将java对象的基本数据类型和图形写入OutputStream ObjectInputStream:对以前使用ObjectOutp ...

  2. java对象流与序列化

    Object流,直接把obj写入或读出. 前言: 比如 画图的程序,咣当画一个三角形出来,咣当画一正方形出来.然后存盘,当你下次再打开软件的时候三角形.方块还在原来的位置上.如果用面向对象的思维,三角 ...

  3. 动车上的书摘-java网络 连接服务器

    摘要: 摘要: 原创出处: http://www.cnblogs.com/Alandre/ 泥沙砖瓦浆木匠 希望转载,保留摘要,谢谢! 应该有些延迟,你会看到黑幕中弹出 来自USA的X原子的计量时间: ...

  4. java --对象流与对象的序列化

    对象流 ObjectInputStream ObjectOutputStream类分别是InputStream和OutputStream的子类,对象输出流使用writeObject(Object ob ...

  5. Java对象流的使用

    为了让对象持久化(把对象存储到本地),可以使用java的对象流处理对象,把对象的内容写到本地存储的文件中,也可以从本地文件中读取出来.也就是常说的序列化和反序列化 主要用到了ObjectInputSt ...

  6. java对象流(一)

    注意:字节数组流是可以不用关闭的(字符数组流要不要关闭暂时不清楚). 对象流的读数据和写数据方法分别是writeObject(Object o)和readObject(Object o). Objec ...

  7. java对象流(二)

    对象流,可以将java中的对象转为字节进行输出.将对象写入文件时.文件输出流是将字节写入到文件中. 对象流是将给定的对象转化为一组字节.writeObject()方法就是将对象转为字节. 对象流,读的 ...

  8. java 对象流的简单使用

    对象的输入输出流的作用: 用于写入对象 的信息和读取对象的信息. 使得对象持久化.   ObjectInputStream   : 对象输入流   ObjectOutPutStream  :对象输出流 ...

  9. Java基础IO类之对象流与序列化

    对象流的两个类: ObjectOutputStream:将Java对象的基本数据类型和图形写入OutputStream ObjectInputStream:对以前使用ObjectOutputStrea ...

随机推荐

  1. web前端开发过程遇到的问题及解决方式(一)

    1. CSS样式优先级问题 <style> .className input(可换其他元素){ type:val !important; } </style> <body ...

  2. 小程序跳转tabbar页面

    如果在app.json 配置tabbar 的时候配置了 跳转的页面的链接: 在其余的子页面,设置用navigator 进行跳转会发现 在tabbar 设置过的页面无法进行跳转,这时需要在navigat ...

  3. 字符串、数组、对象常用API

    常用的字符串API  1.常见方法和属性 length 属性,获取字符串的字符数量 charAt(i) 返回给定位置的字符 charCodeAt( ) 返回给定位置的字符的字符编码 <scrip ...

  4. IDEA中通过工具类实现对数据库的增删改查

    package com.hu.dao; import com.hu.entity.Student; import java.util.List;import java.util.Map; public ...

  5. 2019.03.12 codeforces739E. Gosha is hunting(dp凸优化)

    传送门 题意:nnn个物品,有aaa个XXX道具和bbb个YYY道具,XXX道具移走第iii个物品概率为pip_ipi​,YYY道具移走第iii个道具概率为uiu_iui​. 对于每个物品每种道具最多 ...

  6. nodejs + 小程序云函数 生成小程序码

    前言:这个东西坑死我了 业务需求要生成小程序码 然后我找了两天的资料 运行 生成一堆的乱码 死活就是不能生成 最后看了一遍博客 套用了一下 自己又简单的改了一下  nodejs 我是刚刚接触  有很多 ...

  7. require()  module.export    Object.keys()

    import API from"../../api/api.js";   var data = require('../../utils/data.js').songs;   // ...

  8. 【翻译】从头开始写一个时间序列数据库-Writing a Time Series Database from Scratch

    本文来自: https://fabxc.org/tsdb/, 如翻译有误,请纠正. 我是从事监控工作的.特别是Prometheus, 一个包含自定义的时间序列库以及集成Kuberntes的监控系统. ...

  9. 查询树节点、oracle、select...start with...connect by prior...

    通过子节点向根节点追朔. select * from persons.dept start with deptid=76 connect by prior paredeptid=deptid 通过根节 ...

  10. scrollIntoView() 调用元素就可以出现在视窗中

    /* 如果滚动页面也是DOM没有解决的一个问题.为了解决这个问题,浏览器实现了一下方法, 以方便开发人员如何更好的控制页面的滚动.在各种专有方法中,HTML5选择了scrollIntoView() 作 ...