白骑士的Java教学高级篇 5.4 序列化

        欢迎继续学习Java高级篇!在前面的章节中,我们已经探讨了Java中的枚举类型。接下来,我们将讨论Java中的另一项重要特性——序列化。序列化(Serialization)是将对象的状态转换为字节流的过程,这样对象就可以通过网络传输、存储到文件或数据库中。反序列化(Deserialization)则是将字节流恢复为对象的过程。通过学习序列化,你将能够更好地处理对象的持久化和传输问题,增强应用程序的灵活性和可靠性。

序列化的基本概念

        序列化是指将对象转换为字节流的过程,而反序列化是将字节流转换回对象的过程。在Java中,实现序列化的类需要实现‘java.io.Serializable‘接口,这个接口是一个标记接口,没有任何方法。

        示例:

import java.io.*;


public class Person implements Serializable {
    private static final long serialVersionUID = 1L;
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{name='" + name + "', age=" + age + '}';
    }

    public static void main(String[] args) {
        Person person = new Person("Alice", 30);

        // 序列化
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("person.ser"))) {
            oos.writeObject(person);
        } 

        catch (IOException e) {
            e.printStackTrace();
        }

        // 反序列化
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("person.ser"))) {
            Person deserializedPerson = (Person) ois.readObject();
            System.out.println("Deserialized Person: " + deserializedPerson);
        } 

        catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

        在这个示例中,‘Person‘类实现了‘Serializable‘接口,使得对象可以被序列化和反序列化。我们将‘Person‘对象序列化到文件‘person.ser‘中,并随后从文件中反序列化该对象。

序列化的控制

        在某些情况下,我们可能不希望序列化对象的所有字段。可以使用‘transient‘关键字来标记不需要序列化的字段。

        示例:

import java.io.*;


public class Employee implements Serializable {
    private static final long serialVersionUID = 1L;
    private String name;
    private transient int ssn; // 社会安全号码不需要序列化

    public Employee(String name, int ssn) {
        this.name = name;
        this.ssn = ssn;
    }

    @Override
    public String toString() {
        return "Employee{name='" + name + "', ssn=" + ssn + '}';
    }

    public static void main(String[] args) {
        Employee employee = new Employee("Bob", 123456789);

        // 序列化
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("employee.ser"))) {
            oos.writeObject(employee);
        } 

        catch (IOException e) {
            e.printStackTrace();
        }

        // 反序列化
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("employee.ser"))) {
            Employee deserializedEmployee = (Employee) ois.readObject();
            System.out.println("Deserialized Employee: " + deserializedEmployee);
        } 

        catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

        在这个示例中,‘ssn‘字段被标记为‘transient‘,因此它不会被序列化。在反序列化过程中,‘ssn‘字段的值将会是默认值(整数的默认值为0)。

序列化的自定义

        有时候,我们需要对序列化过程进行更多的控制。可以通过实现‘writeObject‘和‘readObject‘方法来自定义序列化和反序列化的过程。

        示例:

import java.io.*;


public class Car implements Serializable {
    private static final long serialVersionUID = 1L;
    private String model;
    private int year;

    public Car(String model, int year) {
        this.model = model;
        this.year = year;
    }

    private void writeObject(ObjectOutputStream oos) throws IOException {
        oos.defaultWriteObject();
        oos.writeInt(year + 1900); // 自定义年份的序列化
    }

    private void readObject(ObjectInputStream ois) throws IOException, ClassNotFoundException {
        ois.defaultReadObject();
        year = ois.readInt() - 1900; // 自定义年份的反序列化
    }

    @Override
    public String toString() {
        return "Car{model='" + model + "', year=" + year + '}';
    }

    public static void main(String[] args) {
        Car car = new Car("Tesla", 2023);

        // 序列化
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("car.ser"))) {
            oos.writeObject(car);
        } 

        catch (IOException e) {
            e.printStackTrace();
        }

        // 反序列化
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("car.ser"))) {
            Car deserializedCar = (Car) ois.readObject();
            System.out.println("Deserialized Car: " + deserializedCar);
        } 

        catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

        在这个示例中,我们通过实现‘writeObject‘和‘readObject‘方法来自定义了‘Car‘对象的序列化和反序列化过程。

序列化的应用场景

        序列化在Java中的应用非常广泛,常见的应用场景包括:

  • 持久化:将对象的状态保存到文件或数据库中,以便在以后恢复。
  • 网络传输:在分布式系统中,通过网络传输对象的状态。
  • 缓存:将对象缓存到内存中,以便在需要时快速恢复。

        示例:

import java.io.*;
import java.util.HashMap;
import java.util.Map;


public class Cache implements Serializable {
    private static final long serialVersionUID = 1L;
    private Map<String, String> data;

    public Cache() {
        this.data = new HashMap<>();
    }

    public void put(String key, String value) {
        data.put(key, value);
    }

    public String get(String key) {
        return data.get(key);
    }

    public static void main(String[] args) {
        Cache cache = new Cache();
        cache.put("key1", "value1");
        cache.put("key2", "value2");

        // 序列化
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("cache.ser"))) {
            oos.writeObject(cache);
        } 

        catch (IOException e) {
            e.printStackTrace();
        }

        // 反序列化
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("cache.ser"))) {
            Cache deserializedCache = (Cache) ois.readObject();
            System.out.println("Deserialized Cache: key1=" + deserializedCache.get("key1"));
            System.out.println("Deserialized Cache: key2=" + deserializedCache.get("key2"));
        } 

        catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

        在这个示例中,我们定义了一个简单的缓存类‘Cache‘,并演示了如何序列化和反序列化缓存对象。

总结

        在本篇博客中,我们详细介绍了Java中的序列化。从序列化的基本概念、序列化的控制、自定义序列化,到序列化的应用场景,通过这些知识的学习,你将能够更好地处理对象的持久化和传输问题,增强应用程序的灵活性和可靠性。序列化是Java编程中非常重要的技术,掌握这些技巧能够帮助你开发高效且灵活的应用程序。在接下来的学习中,我们将继续探讨Java高级篇中的其他重要内容,帮助你全面掌握Java编程技能。祝你学习愉快,不断进步!

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

白骑士所长

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值