动车上的书摘-java对象流与序列化

动车上的书摘-java对象流与序列化

一.对象序列化

当需要存储相同类型的数据,选择固定的长度记录是好选择。但是在面向对象(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)保存的。如图(两个管理拥有同一个工作人员):

Unnamed QQ Screenshot20140717164755

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

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

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

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

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

 

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

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

 

三.序列化案例

下面,我们就依照上面的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 02 00:00:00 CST 1993]
Employee:Manage [worker=Employee [name=Employee one, salary=5000.0, hireDay=Sun May 02 00:00:00 CST 1993]]
Employee:Manage [worker=Employee [name=Employee one, salary=5000.0, hireDay=Sun May 02 00:00:00 CST 1993]]

java.io.ObjectOutputStream

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

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

 

java.io.ObjectInputStream

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

readObject()
          从 ObjectInputStream 读取对象。

四.感谢知识来源和小结

一.对象序列化

二.对象序列化中的算法

三.序列化案例

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/109046.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • yourphp常用标签

    yourphp常用标签

  • C语言必背18个经典程序,2022年C语言必背100代码大全

    C语言必背18个经典程序,2022年C语言必背100代码大全对于c语言来说,要记得东西其实不多,基本就是几个常用语句加一些关键字而已。你所看到的那些几千甚至上万行的代码,都是用这些语句和关键词来重复编写的。只是他们逻辑功能不一样,那如何快速的上手C语言代码,建议多看多写,下面是小编整理的C语言必背18个经典程序。

  • tar 解压命令

    tar 解压命令

  • 深度残差网络(ResNet)之ResNet34的实现和个人浅见[通俗易懂]

    深度残差网络(ResNet)之ResNet34的实现和个人浅见[通俗易懂]残差网络是由来自MicrosoftResearch的4位学者提出的卷积神经网络,在2015年的ImageNet大规模视觉识别竞赛(ImageNetLargeScaleVisualRecognitionChallenge,ILSVRC)中获得了图像分类和物体识别的优胜。**残差网络的特点是容易优化,并且能够通过增加相当的深度来提高准确率。其内部的残差块使用了跳跃连接(shortcut),缓解了在深度神经网络中增加深度带来的梯度消失问题**。残差网络(ResNet)的网络结构图举例如下:

  • Redmi路由器AC2100之Openwrt旁路由设置

    Redmi路由器AC2100之Openwrt旁路由设置一、思路和环境:1、Redmi路由器AC2100作为主路由,路由系统为Padavan,主要负责拨号、DHCP、WIFI等功能,网络地址为192.168.11.1。2、群晖虚拟机vmm安装koolshare的Openwrt,单臂软路由作为旁路由,以实现zerotier等插件功能,网络地址为192.168.11.11。3、将旁路由的网关指向主路由网络地址192.168.11.1,关闭DHCP和桥接;同时,将主路由的网关指向旁路由的网络地址192.168.11.11。通过主路由和旁路由互指网关实现。二、旁路由设置

  • 『迷你教程』机器学习的Bootstrap及Python实现[通俗易懂]

    『迷你教程』机器学习的Bootstrap及Python实现[通俗易懂]文章目录引导法引导程序的配置引导程序APIBootstrap方法是一种重采样技术,用于通过对数据集进行替换采样来估计总体统计数据。它可用于估计汇总统计数据,例如均值或标准差。它在应用机器学习中用于在对未包含在训练数据中的数据进行预测时估计机器学习模型的技能。估计机器学习模型技能的结果的一个理想特性是可以用置信区间表示估计的技能,这是其他方法(例如交叉验证)不容易获得的特征。在本文中您将发现用于估计机器学习模型对未知数据的技能的引导重采样方法。bootstrap方法涉及对数据集进行迭代重采样

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号