0

0

Java中对象流怎么使用 掌握Java序列化对象的读写方法

冰火之心

冰火之心

发布时间:2025-06-22 13:21:02

|

826人浏览过

|

来源于php中文网

原创

java对象流用于序列化和反序列化,即将对象转换为字节流以实现存储或传输。1. 要实现序列化,类需实现serializable接口并建议显式声明serialversionuid;2. 使用objectoutputstream将对象写入输出流完成序列化;3. 使用objectinputstream从输入流读取对象完成反序列化,需强制类型转换并处理classnotfoundexception;4. transient关键字标记的字段不会被序列化,反序列化后值为默认值;5. 可通过自定义writeobject()和readobject()方法实现个性化序列化逻辑;6. 应用场景包括数据持久化、分布式系统通信、缓存及会话管理;7. 风险包括安全漏洞、性能消耗及版本兼容问题,应避免反序列化不可信数据并选择高效框架以优化性能。

Java中对象流怎么使用 掌握Java序列化对象的读写方法

Java中对象流主要用于序列化和反序列化Java对象,简单来说,就是把对象转换成字节流,可以存储到磁盘或者通过网络传输,然后再把字节流转换回对象。这在很多场景下都很有用,比如持久化数据,或者在分布式系统中传递对象。

Java中对象流怎么使用 掌握Java序列化对象的读写方法

序列化和反序列化。

Java中对象流怎么使用 掌握Java序列化对象的读写方法

如何实现Java对象的序列化?

要实现Java对象的序列化,首先需要让你的类实现 java.io.Serializable 接口。这个接口是一个标记接口,没有任何方法需要实现,它的作用仅仅是告诉JVM,这个类的对象是可以被序列化的。

立即学习Java免费学习笔记(深入)”;

Java中对象流怎么使用 掌握Java序列化对象的读写方法
import java.io.Serializable;

public class MyObject implements Serializable {

    private static final long serialVersionUID = 1L; // 建议显示的声明 serialVersionUID

    private String name;
    private int age;

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

    // getter 和 setter 方法
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

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

注意 serialVersionUID 的声明。 serialVersionUID 是序列化版本号,用于在反序列化时验证类的版本一致性。 如果类结构发生改变,但 serialVersionUID 没有改变,反序列化可能会失败。 建议显式声明,避免JVM自动生成带来的潜在问题。

接下来,使用 ObjectOutputStream 将对象写入到输出流中:

import java.io.*;

public class SerializationExample {

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

        try (FileOutputStream fileOutputStream = new FileOutputStream("myobject.ser");
             ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream)) {

            objectOutputStream.writeObject(myObject);
            System.out.println("对象已序列化");

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

这里使用了try-with-resources语句,确保流在使用完毕后会被自动关闭。

如何反序列化Java对象?

反序列化就是将字节流转换回对象。 使用 ObjectInputStream 从输入流中读取对象:

import java.io.*;

public class DeserializationExample {

    public static void main(String[] args) {
        try (FileInputStream fileInputStream = new FileInputStream("myobject.ser");
             ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream)) {

            MyObject myObject = (MyObject) objectInputStream.readObject();
            System.out.println("对象已反序列化: " + myObject);

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

注意, readObject() 方法返回的是 Object 类型,需要强制类型转换成目标类型。 另外,还需要处理 ClassNotFoundException,因为反序列化时可能会找不到对应的类定义。

GradPen论文
GradPen论文

GradPen是一款AI论文智能助手,深度融合DeepSeek,为您的学术之路保驾护航,祝您写作顺利!

下载

序列化时遇到 transient 关键字会发生什么?

如果在类的字段上使用了 transient 关键字,那么在序列化时,这个字段的值会被忽略。 也就是说,在反序列化后,这个字段的值会是默认值(例如,int 类型的默认值是0,String 类型的默认值是null)。

import java.io.Serializable;

public class MyObjectWithTransient implements Serializable {

    private static final long serialVersionUID = 2L;

    private String name;
    private transient int age; // age 字段被标记为 transient

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

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

在序列化和反序列化 MyObjectWithTransient 对象后,age 字段的值会变成0。 transient 关键字通常用于标记那些不应该被持久化或者不方便持久化的字段,例如密码或者敏感信息。

如何自定义序列化和反序列化过程?

如果默认的序列化和反序列化过程不能满足需求,可以自定义序列化和反序列化过程。 在类中实现 writeObject()readObject() 方法,这两个方法会在序列化和反序列化时被自动调用。

import java.io.*;

public class MyObjectWithCustomSerialization implements Serializable {

    private static final long serialVersionUID = 3L;

    private String name;
    private int age;

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    private void writeObject(ObjectOutputStream out) throws IOException {
        // 自定义序列化逻辑
        out.writeObject("Custom: " + name);
        out.writeInt(age + 100); // 为了演示,将 age 加 100
    }

    private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
        // 自定义反序列化逻辑
        name = (String) in.readObject();
        age = in.readInt() - 100; // 恢复 age 的原始值
    }

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

writeObject() 方法中,可以自定义序列化逻辑,例如加密敏感数据或者压缩数据。 在 readObject() 方法中,需要按照 writeObject() 方法的顺序读取数据,并进行相应的处理。 需要注意的是,这两个方法的签名必须是 private,并且抛出 IOExceptionClassNotFoundException 异常。

序列化在实际项目中有哪些应用场景?

序列化在实际项目中有很多应用场景:

  • 持久化数据: 将对象存储到磁盘或者数据库中,以便以后使用。
  • 分布式系统: 在不同的JVM之间传递对象,例如在RPC框架中。
  • 缓存: 将对象存储到缓存中,提高访问速度。
  • 会话管理: 在Web应用中,将用户的会话信息存储到服务器上。

例如,在Spring Session中,可以使用序列化将用户的会话信息存储到Redis或者其他存储介质中,实现会话共享。 另外,在Dubbo等RPC框架中,也需要使用序列化将请求和响应对象在不同的服务提供者和消费者之间传递。

序列化有哪些潜在的风险和注意事项?

序列化虽然很方便,但也存在一些潜在的风险和注意事项:

  • 安全风险: 反序列化可能会导致安全漏洞,例如反序列化漏洞。 攻击者可以构造恶意的序列化数据,利用反序列化过程执行任意代码。 为了避免这种风险,应该尽量避免反序列化不受信任的数据,或者使用安全的序列化框架。
  • 性能问题: 序列化和反序列化会消耗一定的CPU和内存资源。 对于大型对象或者高并发场景,可能会影响性能。 可以考虑使用更高效的序列化框架,例如Protobuf或者Kryo。
  • 版本兼容性: 如果类的结构发生改变,可能会导致反序列化失败。 应该尽量保持类的结构稳定,或者使用 serialVersionUID 来保证版本兼容性。

总的来说,Java对象流是处理对象序列化的重要工具,理解其使用方法和注意事项,可以帮助你更好地在实际项目中应用序列化技术。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

114

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

29

2026.01.26

什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

328

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

235

2023.10.07

dubbo和zookeeper有什么区别
dubbo和zookeeper有什么区别

dubbo和zookeeper的区别:1、功能定位;2、使用场景;3、数据存储与协调;4、集成与关系;5、性能与可靠性;6、扩展性与灵活性;7、社区与生态系统。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

226

2024.02.23

springcloud和dubbo有哪些区别
springcloud和dubbo有哪些区别

springcloud和dubbo的区别:1、定位与关注点;2、生态环境与集成性;3、调用方式与性能;4、组件与功能;5、定制性与灵活性;6、学习曲线与上手难度;7、社区支持与维护。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

123

2024.02.23

dubbo原理和机制是什么
dubbo原理和机制是什么

dubbo原理和机制的解释:1、核心组件;2、通信原理;3、集群容错;4、自动发现与注册;5、负载均衡与路由;6、序列化与传输;7、监控与日志;8、扩展性;9、安全性;10、与spring集成等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

98

2024.02.23

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

1

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.9万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号