【Android】使用 Intent 传递对象的两种序列化方式

【Android】使用 Intent 传递对象的两种序列化方式

Android 中我们经常使用 Intent 在不同的组件之间传递数据,比如从一个 Activity 跳转到另一个 Activity 时传递参数。对于简单的数据(如 intString 等),使用 putExtra 非常方便。但如果我们要传递 自定义对象,就需要使用更复杂的方式。

方式一:Serializable

1. 定义对象类

java 复制代码
import java.io.Serializable;

public class User implements Serializable {
    private String name;
    private int age;

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

    // Getter / Setter 省略
}

2. 发送方 Activity

java 复制代码
Intent intent = new Intent(this, SecondActivity.class);
User user = new User("张三", 20);
intent.putExtra("user", user);
startActivity(intent);

3. 接收方 Activity

java 复制代码
User user = (User) getIntent().getSerializableExtra("user");

这里调用了 getSerializableExtra() 方法来获取通过参数传递过来的序列化对象,接着再将它向下转型成 Person 对象,这样我们就成功实现了使用 Intent 来传递对象的功能了。

原理 :通过 Java 的 ObjectOutputStream 将对象转为字节流,反序列化时用 ObjectInputStream 重建对象。

优点

  • 实现简单,只需实现 Serializable 接口。

缺点

  • 性能较差。
  • 不适合频繁传输或大对象。
  • 不能控制序列化过程,不安全。

方式二:Parcelable

Android 官方推荐使用 Parcelable,因为它序列化/反序列化速度更快,内存使用更高效。

1. 定义对象类

java 复制代码
import android.os.Parcel;
import android.os.Parcelable;

public class User implements Parcelable {
    private String name;
    private int age;

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

    protected User(Parcel in) {
        name = in.readString();
        age = in.readInt();
    }

    public static final Creator<User> CREATOR = new Creator<User>() {
        @Override
        public User createFromParcel(Parcel in) {
            return new User(in);
        }

        @Override
        public User[] newArray(int size) {
            return new User[size];
        }
    };

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(name);
        dest.writeInt(age);
    }

    @Override
    public int describeContents() {
        return 0;
    }

    // Getter / Setter 可选
}

Parcelable 的实现方式要复杂一些,首先让 User 类实现了 Parcelable 接口,这样就必须重写 describeContents()writeToParcel() 这两个方法,其中 describeContents() 方法直接返回0就可以了,而 writeToParcel() 方法中我们需要调用 Parcel 的 writeXxx() 方法,将 Person 类中的字段一一写出。注意,字符串型数据就调用 writestring() 方法,整型数据就调用 writeInt() 方法,以此类推。

关键注意事项

  • createFromParcel() 方法中,字段读取顺序( readString()readInt() )必须严格匹配 writeToParcel() 方法中的字段写入顺序。

  • newArray() 方法只需简单创建对应大小的数组。

2. 发送方 Activity

java 复制代码
Intent intent = new Intent(this, SecondActivity.class);
User user = new User("李四", 25);
intent.putExtra("user", user);
startActivity(intent);

3. 接收方 Activity

java 复制代码
User user = getIntent().getParcelableExtra("user");

原理 :通过 Parcel 内存容器直接操作二进制数据,避免反射开销。

优点

  • 性能优于 Serializable,可自定义序列化过程。

缺点

  • 编写代码繁琐,对于嵌套对象或集合,需要额外实现嵌套类的 Parcelable

两种方式对比

Serializable Parcelable
实现复杂度 简单(只需 implements) 较高(手动写入字段)
序列化速度 快(无反射)
GC/内存开销 大(使用反射) 小(结构紧凑)
是否可控 不可控 精确控制
嵌套支持 自动递归序列化 手动嵌套写入/读取
集合支持 支持所有实现 Serializable 的集合 需使用专门 API,如 writeTypedList

虽然 Serializable 使用简单,但在 Android 中,Parcelable 是更推荐的选择,尤其是在性能敏感或大对象频繁传输的场景。对于有嵌套对象或集合的类:

  • Serializable 可以省心地一键传输;
  • Parcelable 则更灵活、更高效,但需要写更多代码。
相关推荐
ayt0071 分钟前
Netty AbstractNioChannel源码深度剖析:NIO Channel的抽象实现
java·数据库·网络协议·安全·nio
Gofarlic_OMS2 分钟前
装备制造企业Fluent许可证成本分点典型案例
java·大数据·开发语言·人工智能·自动化·制造
程序员雷欧10 分钟前
大模型应用开发学习第八天
大数据·人工智能·学习
码王吴彦祖11 分钟前
顶象 AC 纯算法迁移实战:从补环境到纯算的完整拆解
java·前端·算法
SccTsAxR16 分钟前
算法基石:手撕离散化、递归与分治
c++·经验分享·笔记·算法
晓晓hh31 分钟前
JavaSE学习——set集合和Map映射
学习
开心码农1号1 小时前
Java rabbitMQ如何发送、消费消息、全套可靠方案
java·rabbitmq·java-rabbitmq
西梅汁1 小时前
C++ 观察者模式
笔记
蜡台1 小时前
JetBrains IDEA 安装 卸载相关总结
java·ide·intellij-idea·注册码
WJLSH1231 小时前
TomCat
java·tomcat