序列化原因

序列化的原因基本可以归纳为以下三种情况:

  1. 永久性保存对象,保存对象的字节序列到本地文件中;
  2. 对象在网络中传递;
  3. 对象在IPC间传递。

序列化方法

在Android系统中关于序列化的方法一般有两种,分别是实现Serializable接口和Parcelable接口,其中Serializable接口是来自Java中的序列化接口,而Parcelable是Android自带的序列化接口。

上述的两种序列化接口都有各自不同的优缺点,我们在实际使用时需根据不同情况而定。

  1. Serializable在序列化的时候会产生大量的临时变量,从而引起频繁的GC,而相比之下Parcelable的性能更高(毕竟是Android自带的),所以当在使用内存时(如:序列化对象在网络中传递对象或序列化在进程间传递对象),更推荐使用Parcelable接口。
  2. 但Parcelable有个明显的缺点:不能能使用在要将数据存储在磁盘上的情况(如:永久性保存对象,保存对象的字节序列到本地文件中),因为Parcel本质上为了更好的实现对象在IPC间传递并不是一个通用的序列化机制,当改变任何Parcel中数据的底层实现都可能导致之前的数据不可读取,所以此时还是建议使用Serializable 。

代码实现

Serializable接口的实现及使用

  • Serializable的接口实现很简单,只需让需要序列化的类继承Serializable 即可,系统会自动将其序列化,具体代码如下:

     public class Book implements Serializable {
    private static final long serialVersionUID = 21455356667888L;
    private String mName;
    private String mPrice; public String getmName() {
    return mName;
    } public void setmName(String mName) {
    this.mName = mName;
    } public String getmPrice() {
    return mPrice;
    } public void setmPrice(String mPrice) {
    this.mPrice = mPrice;
    } }
  • 在Activity中使用方法:

     // serializable对象传递方法
    public void setSerializableMethod() {
    Book book = new Book();
    book.setmName("王海康");
    book.setmPrice("20$");
    Intent intent = new Intent(this, BookTest.class);
    Bundle bundle = new Bundle();
    bundle.putSerializable(SER_KEY, book);
    intent.putExtras(bundle);
    startActivity(intent);
    } // serializable对象获取方法
    public Book getSerializableMethod(){
    Book mBook = (Book )getIntent().getSerializableExtra(SER_KEY);
    return mBook;
    }

Parcelable接口的实现及使用

  • 实现Parcelable接口主要可以分为一下几步:

    1)implements Parcelable。

    2)重写writeToParcel方法,将你的对象序列化为一个Parcel对象,即:将类的数据写入外部提供的Parcel中,打包需要传递的数据到Parcel容器保存,以便从Parcel容器获取数据。

    3)重写describeContents方法,内容接口描述,默认返回0即可。

    4)实例化静态内部对象CREATOR实现接口Parcelable.Creator 。

    注意:若将Parcel看成是一个流,则先通过writeToParcel把对象写到流里面,再通过createFromParcel从流里读取对象,因此类实现的写入顺序和读出顺序必须一致。

    具体实现代码如下:

     public class Person implements Parcelable {
    private String mName;
    private String mSex;
    private int mAge; public String getmName() {
    return mName;
    } public void setmName(String mName) {
    this.mName = mName;
    } public String getmSex() {
    return mSex;
    } public void setmSex(String mSex) {
    this.mSex = mSex;
    } public int getmAge() {
    return mAge;
    } public void setmAge(int mAge) {
    this.mAge = mAge;
    } @Override
    public int describeContents() {
    return 0;
    } @Override
    public void writeToParcel(Parcel dest, int flags) {
    dest.writeString(mName);
    dest.writeString(mSex);
    dest.writeInt(mAge);
    } public static final Parcelable.Creator<Person> CREATOR = new Creator<Person>() { @Override
    public Person createFromParcel(Parcel source) {
    Person person = new Person();
    person.mName = source.readString();
    person.mSex = source.readString();
    person.mAge = source.readInt();
    return person;
    } //供反序列化本类数组时调用的
    @Override
    public Person[] newArray(int size) {
    return new Person[size];
    }
    };
  • 在Activity中使用方法:

    1)传递单一对象,具体代码如下:

     // parcelable对象传递方法
    public void setParcelableMethod() {
    Person person = new Person();
    person.setmName("王海康");
    person.setmSex("男");
    person.setmAge(45);
    Intent intent = new Intent(this, PersonTest.class);
    Bundle bundle = new Bundle();
    bundle.putParcelable(PAR_KEY, person);
    intent.putExtras(bundle);
    startActivity(intent);
    } // parcelable对象获取方法
    public Person getParcelableMethod(){
    Person mPerson = (Person)getIntent().getParcelableExtra(PAR_KEY);
    return mPerson;
    }

    2)传递对象列表,具体代码如下:

    需要注意的是,若List personList = new ArrayList();则会报错,因为下面调用的putParcelableArrayList()函数其中一个参数的类型为ArrayList。

     // parcelable对象List传递方法
    public void setParcelableListMethod() {
    ArrayList<Person> personList = new ArrayList<Person>();
    Person person1 = new Person();
    person1.setmName("王海康");
    person1.setmSex("男");
    person1.setmAge(45);
    personList.add(person1);
    Person person2 = new Person();
    person2.setmName("薛岳");
    person2.setmSex("男");
    person2.setmAge(15);
    personList.add(person2);
    Intent intent = new Intent(this, PersonTest.class);
    Bundle bundle = new Bundle();
    bundle.putParcelableArrayList(PAR_LIST_KEY, personList);
    intent.putExtras(bundle);
    startActivity(intent);
    } // parcelable对象获取方法
    public ArrayList<Person> getParcelableMethod(){
    ArrayList<Person> mPersonList = getIntent().getParcelableExtra(PAR_LIST_KEY);
    return mPersonList;
    }

    3)最后介绍一个投机取巧的方法:

    不用继承Parcelable或Serializable方法即可实现IPC中对象的传递。这种方法的实现原理不是很明白,只知道代码中new ArrayList()返回的其实是一个EmptyArray.OBJECT数组,不过我感觉应该还是系统调用Serializable进行序列化的,如果各位读者有好的想法,欢迎告知。

    具体代码如下:

     //对象List传递
    public void setObjectMethod(){
    ......(省略)
    ArrayList list = new ArrayList();
    //ObjectList为某一对象列表
    list.add(ObjectList);
    bundle.putParcelableArrayList(PAR_LIST_KEY, list);
    intent.putExtras(bundle);
    startActivity(intent);
    } //获取对象List
    ArrayList list = bundle.getParcelableArrayList("list");
    //强转成你自己定义的list,这样ObjectList就是你传过来的那个list了。
    ObjectList= (List<Object>) list.get(0);

参考资料:

1.Android Parcel官方文档

2.Android Serializable 和 Parcelable 区别

3.Android中Parcelable接口用法

查看原文:http://www.xyczero.com/blog/article/8/.

浅谈Android序列化的更多相关文章

  1. 浅谈Android保护技术__代码混淆

    浅谈Android保护技术__代码混淆   代码混淆 代码混淆(Obfuscated code)亦称花指令,是将计算机程序的代码,转换成一种功能上等价,但是难于阅读和理解的形式的行为.将代码中的各种元 ...

  2. 浅谈Android应用保护(一):Android应用逆向的基本方法

    对于未进行保护的Android应用,有很多方法和思路对其进行逆向分析和攻击.使用一些基本的方法,就可以打破对应用安全非常重要的机密性和完整性,实现获取其内部代码.数据,修改其代码逻辑和机制等操作.这篇 ...

  3. 安卓开发_浅谈Android动画(四)

    Property动画 概念:属性动画,即通过改变对象属性的动画. 特点:属性动画真正改变了一个UI控件,包括其事件触发焦点的位置 一.重要的动画类及属性值: 1.  ValueAnimator 基本属 ...

  4. 浅谈Android应用性能之内存

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 文/ jaunty [博主导读]在Android开发中,不免会遇到许多OOM现象,一方面可能是由于开 ...

  5. 浅谈Android五大布局

    Android的界面是有布局和组件协同完成的,布局好比是建筑里的框架,而组件则相当于建筑里的砖瓦.组件按照布局的要求依次排列,就组成了用户所看见的界面.Android的五大布局分别是LinearLay ...

  6. [转]浅谈Android五大布局(二)——RelativeLayout和TableLayout

    在浅谈Android五大布局(一)中已经描述了LinearLayout(线性布局).FrameLayout(单帧布局)和AbsoulteLayout(绝对布局)三种布局结构,剩下的两种布局Relati ...

  7. [转]浅谈Android五大布局(一)——LinearLayout、FrameLayout和AbsoulteLayout

    Android的界面是有布局和组件协同完成的,布局好比是建筑里的框架,而组件则相当于建筑里的砖瓦.组件按照布局的要求依次排列,就组成了用户所看见的界面.Android的五大布局分别是LinearLay ...

  8. 浅谈Android Studio3.0更新之路(遇坑必入)

    >可以参考官网设置-> 1 2 >> Fantasy_Lin_网友评论原文地址是:简书24K纯帅豆写的我也更新一下出处[删除]Fa 转自脚本之家 浅谈Android Studi ...

  9. 浅谈android代码保护技术_ 加固

    浅谈android代码保护技术_加固 导语 我们知道Android中的反编译工作越来越让人操作熟练,我们辛苦的开发出一个apk,结果被人反编译了,那心情真心不舒服.虽然我们混淆,做到native层,但 ...

随机推荐

  1. Xshell中文编码的设置

    一直用的是SSH Secure Shell Client,关于中文乱码,一直找不到简便的解决方案,后来改用XShell,编码设置如下: 1.查看linux系统编码,linux命令: locale. 2 ...

  2. [Locked] Flatten 2D Vector

    Problem Description: Implement an iterator to flatten a 2d vector. For example,Given 2d vector = [ [ ...

  3. lightoj 1291 无向图边双联通+缩点统计叶节点

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1291 #include<cstdio> #include<cstri ...

  4. Linux 输出重定向>和>>的区别是什么

    > 是定向输出到文件,如果文件不存在,就创建文件:如果文件存在,就将其清空:一般我们备份清理日志文件的时候,就是这种方法:先备份日志,再用`>`,将日志文件清空(文件大小变成0字节): & ...

  5. Hard 随机洗牌函数 @CareerCup

    第i个元素和index在[i,length-1]之间的一个数随机交换 package Hard; import CtCILibrary.AssortedMethods; /** * * Write a ...

  6. Android 更新UI的两个方法

    Android 更新UI的两个方法 在Android的开发过程中,常常需要适时的更新UI.Androd中的UI是在主线程中更新的.如果在主线程之外的线程中直接更新,就会出现报错并抛出异常: andro ...

  7. POJ 2049 Finding Nemo

    Finding Nemo Time Limit: 2000MS   Memory Limit: 30000K Total Submissions: 8631   Accepted: 2019 Desc ...

  8. sublime text3 插件配置

    (转) sublme text 全程指引:http://www.cnblogs.com/figure9/p/sublime-text-complete-guide.html 使用Package Con ...

  9. Linux开发工具之gdb(上)

    三.gdb调试(上) 01.gdb:gdb是GNU debugger的缩写,是编程调试工作. 功能:   启动程序,可以按照用户自定义的要求随心所欲的运行程序:   可让被调试的程序在用户所指定的调试 ...

  10. BeanFactory学习

    关于BeanFactory,一个相对易懂的博客,关于深入的理解,后续继续学习 http://www.cnblogs.com/liuling/archive/2013/04/14/BeanFactory ...