流-为序列化解释

流(Stream)是什么?

流(Stream)是 Java 中用来处理输入和输出(I/O)操作的基本抽象,它可以用来读取和写入数据。流分为两种:

  • 字节流(Byte Stream): 用于处理原始二进制数据,如文件、图片等。

  • 字符流(Character Stream): 用于处理字符数据,支持对字符数据的编码和解码。

在序列化和反序列化的过程中,我们主要使用 字节流 。特别是在将对象写入到文件、网络或数据库时,会用到 ObjectOutputStreamObjectInputStream 这两个类,它们分别属于字节流的一部分。

什么时候会用到流?

在实际开发中,流常用于以下几个场景:

1. 文件操作:
  • 读取和写入文件: 在开发中,常常需要将对象保存到文件中,或者从文件中读取对象数据。这时我们就会用到流,结合序列化来实现对象的持久化。

示例: 将对象写入文件:

复制代码
import java.io.*;

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

        // 序列化:将对象写入文件
        FileOutputStream fileOut = new FileOutputStream("person.ser");
        ObjectOutputStream out = new ObjectOutputStream(fileOut);
        out.writeObject(person);  // 对象序列化到文件
        out.close();
        fileOut.close();

        // 反序列化:从文件读取对象
        FileInputStream fileIn = new FileInputStream("person.ser");
        ObjectInputStream in = new ObjectInputStream(fileIn);
        Person deserializedPerson = (Person) in.readObject();
        in.close();
        fileIn.close();

        System.out.println(deserializedPerson);
    }
}

在这个例子中,ObjectOutputStream 用来将 person 对象序列化并写入文件,而 ObjectInputStream 用来从文件中读取并反序列化回原始对象。

2. 网络传输:
  • 客户端与服务器之间的数据传输: 在分布式系统中,客户端和服务器之间需要通过网络传输数据。如果传输的是对象,那么对象必须先序列化为字节流,才能通过网络发送。接收端收到数据后再反序列化还原为对象。

示例: 假设有一个简单的客户端和服务器,客户端将对象发送给服务器:

复制代码
// 客户端发送对象
Socket socket = new Socket("localhost", 8080);
ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
Person person = new Person("Bob", 25);
out.writeObject(person);  // 序列化对象并发送
out.close();
socket.close();

// 服务器接收对象
ServerSocket serverSocket = new ServerSocket(8080);
Socket clientSocket = serverSocket.accept();
ObjectInputStream in = new ObjectInputStream(clientSocket.getInputStream());
Person receivedPerson = (Person) in.readObject();  // 反序列化对象
System.out.println("Received person: " + receivedPerson);
in.close();
clientSocket.close();
serverSocket.close();

这个例子演示了如何通过 ObjectOutputStream 将一个对象通过网络发送给服务器,并使用 ObjectInputStream 从网络读取并反序列化对象。

3. 数据库存储(持久化)
  • 对象持久化: 在一些情况下,我们需要将对象的数据存储到数据库中。这时候,我们可能会使用流(通过序列化)来将对象转化为二进制数据,并存入数据库的 BLOB 字段(大二进制对象)。当需要读取对象时,再从数据库中获取二进制数据,通过反序列化将其恢复为 Java 对象。

示例: 假设我们需要将一个 Person 对象存储到数据库中的 BLOB 字段:

复制代码
// 将对象序列化并存储到数据库
String sql = "INSERT INTO persons (data) VALUES (?)";
PreparedStatement statement = connection.prepareStatement(sql);

ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream();
ObjectOutputStream out = new ObjectOutputStream(byteArrayOut);
out.writeObject(person);
byte[] personData = byteArrayOut.toByteArray();

statement.setBytes(1, personData);
statement.executeUpdate();

读取时,我们可以通过 ObjectInputStream 将字节流反序列化回对象:

复制代码
// 从数据库中读取对象并反序列化
String sql = "SELECT data FROM persons WHERE id = ?";
PreparedStatement statement = connection.prepareStatement(sql);
ResultSet rs = statement.executeQuery();

if (rs.next()) {
    byte[] personData = rs.getBytes("data");
    ByteArrayInputStream byteArrayIn = new ByteArrayInputStream(personData);
    ObjectInputStream in = new ObjectInputStream(byteArrayIn);
    Person person = (Person) in.readObject();
    System.out.println(person);
}
4. 会话管理:
  • 在 Web 应用中,使用 HttpSession 来保存用户的会话信息。在某些情况下(如保存用户对象),你可能需要将 Java 对象序列化成字节流存入会话中,或者从会话中读取并反序列化。

例如,存储用户登录信息:

复制代码
HttpSession session = request.getSession();
User user = new User("john", "password123");
session.setAttribute("user", user);  // 将对象存入会话,自动序列化

当需要使用该信息时,可以从会话中取出对象:

复制代码
User user = (User) session.getAttribute("user");  // 获取并反序列化对象

总结:

流主要用于以下几种场景:

  • 文件操作: 将对象存储到文件或从文件读取。

  • 网络传输: 通过网络发送和接收对象。

  • 数据库存储: 将对象序列化为字节流并存入数据库,或从数据库中读取字节流并反序列化。

  • 会话管理: 在 Web 应用中,将对象存储到用户会话中。

在这些场景中,使用 是为了处理数据的输入输出操作,而 序列化反序列化 则是将 Java 对象转换为字节流的关键技术。

相关推荐
LXS_3572 小时前
Day 18 C++提高 之 STL常用容器(string、vector、deque)
开发语言·c++·笔记·学习方法·改行学it
王琦03183 小时前
Python 函数详解
开发语言·python
胡伯来了3 小时前
13. Python打包工具- setuptools
开发语言·python
小鸡吃米…3 小时前
Python 中的多层继承
开发语言·python
deng-c-f3 小时前
Linux C/C++ 学习日记(53):原子操作(二):实现shared_ptr
开发语言·c++·学习
wanghowie4 小时前
01.07 Java基础篇|函数式编程与语言新特性总览
java·开发语言·面试
Cricyta Sevina4 小时前
Java IO 基础理论知识笔记
java·开发语言·笔记
MyBFuture4 小时前
C#接口与抽象类:关键区别详解
开发语言·c#·visual studio
晨晖24 小时前
简单排序c语言版
c语言·开发语言