序列化原因

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

  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. cobbler客户端重装系统

    已有操作系统的主机通过koan从Cobbler服务器重装系统 1,安装epel rpm -ivh http://dl.fedoraproject.org/pub/epel/6/x86_64/epel- ...

  2. 十大纺织品、布料、面料品牌排名 - 十大品牌 - 中国品牌网 Chinapp.com

    十大纺织品.布料.面料品牌排名 - 十大品牌 - 中国品牌网 Chinapp.com 十大纺织品.布料.面料品牌排名

  3. String Problem - HDU 3374 (kmp+最大最小表示)

    题目大意:有一个字符串长度为N的字符串,这个字符串可以扩展出N个字符串,并且按照顺序编号,比如串  ” SKYLONG “ SKYLONG 1 KYLONGS 2 YLONGSK 3 LONGSKY ...

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

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

  5. Hadoop入门--HDFS(单节点)配置和部署 (一)

    一 配置SSH 下载ssh服务端和客户端 sudo apt-get install openssh-server openssh-client 验证是否安装成功 ssh username@192.16 ...

  6. Badge分析&如何逼死处女座

    Badge分析 所谓Badge,原本是iOS上的一个效果,但是被Android抄的多了,也就成了Android的标配.图就不上了,大家都懂的. 应用icon显示角标实际上是在Launcher中实现的, ...

  7. Another app is currently holding the yum lock; waiting for it to exit... 怎么解决

    Another app is currently holding the yum lock; waiting for it to exit... 怎么解决 这个问题说明你的程序yum程序正在运行,必须 ...

  8. SGU 199 - Beautiful People 最长上升子序列LIS

    要邀请n个人参加party,每个人有力量值strength Si和魅力值 beauty Bi,如果存在两人S i ≤ S j and B i ≥ B j 或者  S i ≥ S j and B i ≤ ...

  9. codevs2622数字序列( 连续子序列最大和O(n)算法)

    /* 算法描述:维护一个s[p]表示累加和 并且更新最大值ans 如果s[p]<0 则从p+1重新累加 证明:设某个区间的起点和终点分别为s t 分两种情况 1.t<p:设s2表示1到s的 ...

  10. 转载:C#中事件的由来

    原文地址 http://www.tracefact.net/CSharp-Programming/Delegates-and-Events-in-CSharp.aspx 感谢博主分享! 我们继续思考转 ...