Binder相关面试总结(五):为什么Activity间传递对象需要序列化

}

public List getMemberNameList() {

return mMemberNameList;

}

public void setMemberNameList(List memberNameList) {

mMemberNameList = memberNameList;

}

}

可以看到实现 Serializable 的实现非常简单,除了实体内容外只要创建一个 serialVersionUID 属性就好。

serialVersionUID

从名字就可以看出来,这个 serialVersionUID ,有些类似我们平时的接口版本号,在运行时这个版本号唯一标识了一个可序列化的类。

也就是说,一个类序列化时,运行时会保存它的版本号,然后在反序列化时检查你要反序列化成的对象版本号是否一致,不一致的话就会报错:·InvalidClassException

如果我们不自己创建这个版本号,序列化过程中运行时会根据类的许多特点计算出一个默认版本号。然而只要你对这个类修改了一点点,这个版本号就会改变。这种情况如果发生在序列化之后,反序列化时就会导致上面说的错误。

因此 JVM 规范强烈 建议我们手动声明一个版本号,这个数字可以是随机的,只要固定不变就可以。同时最好是 private 和 final 的,尽量保证不变。

此外,序列化过程中不会保存 static 和 transient 修饰的属性,前者很好理解,因为静态属性是与类管理的,不属于对象状态;而后者则是 Java 的关键字,专门用来标识不序列化的属性。

默认实现 Serializable 不会自动创建 serialVersionUID 属性,为了提示我们及时创建 serialVersionUID ,可以在设置中搜索 serializable 然后选择下图所示的几个选项,为那些没有声明 serialVersionUID 属性的类以及内部类添加一个警告。

在这里插入图片描述

这样当我们创建一个类不声明 UID 属性时,类名上就会有黄黄的警告:

在这里插入图片描述

鼠标放上去就会显示警告内容:

GroupBean’ does not define a ‘serialVersionUID’ field less… (Ctrl+F1)

Reports any Serializable classes which do not provide a serialVersionUID field. Without a serialVersionUID field, any change to a class will make previously serialized versions unreadable.

这时我们按代码提示快捷键就可以生成 serialVersionUID 了。

序列化与反序列化 Serializable

Serializable 的序列化与反序列化分别通过 ObjectOutputStream 和 ObjectInputStream 进行,实例代码如下:

/**

  • 序列化对象

  • @param obj

  • @param path

  • @return

*/

synchronized public static boolean saveObject(Object obj, String path) {

if (obj == null) {

return false;

}

ObjectOutputStream oos = null;

try {

oos = new ObjectOutputStream(new FileOutputStream(path));

oos.writeObject(obj);

oos.close();

return true;

} catch (IOException e) {

e.printStackTrace();

} finally {

if (oos != null) {

try {

oos.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

return false;

}

/**

  • 反序列化对象

  • @param path

  • @param

  • @return

*/

@SuppressWarnings("unchecked ")

synchronized public static T readObject(String path) {

ObjectInputStream ojs = null;

try {

ojs = new ObjectInputStream(new FileInputStream(path));

return (T) ojs.readObject();

} catch (IOException | ClassNotFoundException e) {

e.printStackTrace();

} finally {

close(ojs);

}

return null;

}

Parcelable 接口


Parcelable 是 Android 特有的序列化接口:

public interface Parcelable {

//writeToParcel() 方法中的参数,用于标识当前对象作为返回值返回

//有些实现类可能会在这时释放其中的资源

public static final int PARCELABLE_WRITE_RETURN_VALUE = 0x0001;

//writeToParcel() 方法中的第二个参数,它标识父对象会管理内部状态中重复的数据

public static final int PARCELABLE_ELIDE_DUPLICATES = 0x0002;

//用于 describeContents() 方法的位掩码,每一位都代表着一种对象类型

public static final int CONTENTS_FILE_DESCRIPTOR = 0x0001;

//描述当前 Parcelable 实例的对象类型

//比如说,如果对象中有文件描述符,这个方法就会返回上面的 CONTENTS_FILE_DESCRIPTOR

//其他情况会返回一个位掩码

public int describeContents();

//将对象转换成一个 Parcel 对象

//参数中 dest 表示要写入的 Parcel 对象

//flags 表示这个对象将如何写入

public void writeToParcel(Parcel dest, int flags);

//实现类必须有一个 Creator 属性,用于反序列化,将 Parcel 对象转换为 Parcelable

public interface Creator {

public T createFromParcel(Parcel source);

public T[] newArray(int size);

}

//对象创建时提供的一个创建器

public interface ClassLoaderCreator extends Creator {

//使用类加载器和之前序列化成的 Parcel 对象反序列化一个对象

public T createFromParcel(Parcel source, ClassLoader loader);

}

}

实现了 Parcelable 接口的类在序列化和反序列化时会被转换为 Parcel 类型的数据 。

Parcel 是一个载体,它可以包含数据或者对象引用,然后通过 IBinder 在进程间传递。

实现 Parcelable 接口的类必须有一个 CREATOR 类型的静态变量,下面是一个实例:

public class ParcelableGroupBean implements Parcelable {

private String mName;

private List mMemberNameList;

private User mUser;

/**

  • 需要我们手动创建的构造函数

  • @param name

  • @param memberNameList

  • @param user

*/

public ParcelableGroupBean(String name, List memberNameList, User user) {

mName = name;

mMemberNameList = memberNameList;

mUser = user;

}

/**

  • 1.内容描述

  • @return

*/

@Override

public int describeContents() {

//几乎都返回 0,除非当前对象中存在文件描述符时为 1

return 0;

}

/**

  • 2.序列化

  • @param dest

  • @param flags 0 或者 1

*/

@Override

public void writeToParcel(Parcel dest, int flags) {

dest.writeString(mName);

dest.writeStringList(mMemberNameList);

dest.writeParcelable(mUser, flags);

}

/**

  • 3.反序列化

*/

public static final Creator CREATOR = new Creator() {

/**

  • 反序列创建对象

  • @param in

  • @return

*/

@Override

public ParcelableGroupBean createFromParcel(Parcel in) {

return new ParcelableGroupBean(in);

}

/**

  • 反序列创建对象数组

  • @param size

  • @return

*/

@Override

public ParcelableGroupBean[] newArray(int size) {

return new ParcelableGroupBean[size];

}

};

/**

  • 4.自动创建的的构造器,使用反序列化得到的 Parcel 构造对象

  • @param in

*/

protected ParcelableGroupBean(Parcel in) {

mName = in.readString();

mMemberNameList = in.createStringArrayList();

//反序列化时,如果熟悉也是 Parcelable 的类,需要使用它的类加载器作为参数,否则报错无法找到类

mUser = in.readParcelable(User.class.getClassLoader());

}

}

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
升又不知道该从何学起的朋友,同时减轻大家的负担。**

[外链图片转存中…(img-BrxjeKig-1715503278121)]

[外链图片转存中…(img-zRAy5uwm-1715503278123)]

[外链图片转存中…(img-3itz4hEo-1715503278124)]

[外链图片转存中…(img-YSH7fAiR-1715503278132)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 18
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值