序列化Serializable和Parcelable

版权声明:本文为HaiyuKing原创文章,转载请注明出处!

前言

简单记录下序列化Serializable和Parcelable的使用方法。

Android中Intent如果要传递类对象,可以通过两种方式实现

  • 方式一:Serializable,要传递的类实现Serializable接口传递对象,
  • 方式二:Parcelable,要传递的类实现Parcelable接口传递对象。

Serializable(Java自带):
Serializable是序列化的意思,表示将一个对象转换成可存储或可传输的状态。序列化后的对象可以在网络上进行传输,也可以存储到本地。

Parcelable(android 专用):
除了Serializable之外,使用Parcelable也可以实现相同的效果,
不过不同于将对象进行序列化,Parcelable方式的实现原理是将一个完整的对象进行分解,
而分解后的每一部分都是Intent所支持的数据类型,这样也就实现传递对象的功能了。

实现序列化的作用

1)永久性保存对象,保存对象的字节序列到本地文件中;

2)通过序列化对象在网络中传递对象;

3)通过序列化在进程间传递对象。

选择序列化方法的原则

1)在使用内存的时候,Parcelable比Serializable性能高,所以推荐使用Parcelable。

2)Serializable在序列化的时候会产生大量的临时变量,从而引起频繁的GC。

3)Parcelable不能使用在要将数据存储在磁盘上的情况,因为Parcelable不能很好的保证数据的持续性在外界有变化的情况下。尽管Serializable效率低点,但此时还是建议使用Serializable 。

--摘自《序列化Serializable和Parcelable的理解和区别
 
简而言之:
Parcelable适合用在内存序列化上(也就是用于Intent间的数据传递);
Serializable适合用于将对象序列化到存储设备中或者将对象序列化后通过网络传输(一般用于Socket传输)。

使用方法

新建bean类

1、新建bean类implement Parcelable,重写以下方法

在Android Studio中可以自动引入:

package com.why.project.androidcnblogsdemo.bean;

import android.os.Parcel;
import android.os.Parcelable;

/**
 * Created by HaiyuKing
 * Used 用户bean类
 */

public class UserBean implements Parcelable{
    private String userName;
    private int userId;
    private boolean isVip;
    
    //必须要添加
    public UserBean(){}

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public int getUserId() {
        return userId;
    }

    public void setUserId(int userId) {
        this.userId = userId;
    }

    public boolean isVip() {
        return isVip;
    }

    public void setVip(boolean vip) {
        isVip = vip;
    }

    //默认返回0就可以
    //返回当前对象的内容描述,如果含有文件描述符,返回1,否则返回0,几乎所有情况都返回0
    @Override
    public int describeContents() {
        return 0;
    }

    //将你的对象序列化为一个Parcel对象 即:将类的数据写入外部提供的Parcel中,打包需要传递的数据到Parcel容器保存,以便从 Parcel容器获取数据
    //将当前对象写入序列化结构中,其中flags标识有两种值:0或者1,为1时标识当前对象需要作为返回值返回,不能立即释放资源,几乎所有情况都为0
    @Override
    public void writeToParcel(Parcel parcel, int i) {
        parcel.writeString(userName);
        parcel.writeInt(userId);
        parcel.writeByte((byte) (isVip ? 1 : 0));
    }

    public static final Creator<UserBean> CREATOR = new Creator<UserBean>() {
        //从序列化后的对象中创建原始对象
        @Override
        public UserBean createFromParcel(Parcel in) {
            return new UserBean(in);
        }
        //创建指定长度的原始对象数组
        @Override
        public UserBean[] newArray(int size) {
            return new UserBean[size];
        }
    };
    //从序列化后的对象中创建原始对象
    protected UserBean(Parcel in) {
        userName = in.readString();
        userId = in.readInt();
        isVip = in.readByte() != 0;
    }
}

2、新建bean类implement Serializable,手动指定serialVersionUID的值

不指定serialVersionUID也可以实现序列化,但是应该指定。这个serialVersionUID是用来辅助序列化和反序列化过程的,原则上序列化后的数据中的serialVersionUID只有和当前类的serialVersionUID相同才能够正常地被反序列化。serialVersionUID的详细工作机制是这样的:序列化的时候系统会将当前类的serialVersionUID写入序列化的文件中(也可能是其他中介),当反序列化的时候系统会检测文件中的serialVersionUID,看它是否和当前类的serialVersionUID一致,如果一致就说明序列化的类的版本和当前类的版本是相同的,这个时候可以成功反序列化;否则就说明当前类和序列化的类相比发生了某些变化,比如成员变量的数量、类型可能发生了改变,这个时候是无法正常反序列化的,因此会报错

而手动指定serialVersionUID值的话,序列化和反序列化时两者的serialVersionUID是相同的。如果不手动指定serialVersionUID的值,反序列化时当前类发生了改变,比如增加或者删除成员变量,那么系统就会重新计算当前类的hash值并把赋值给serialVersionUID,这个时候当前类的serialVersionUID就和序列化数据中的serialVersionUID不一致,于是反序列化失败。

--摘自《Android开发艺术探索》

package com.why.project.androidcnblogsdemo.bean;

import java.io.Serializable;

/**
 * Created by HaiyuKing
 * Used 书籍Bean类
 */

public class BookBean implements Serializable {

    private static final long serialVersionUID = 123L;//手动指定值,默认是1L,可以是根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段

    private String bookName;
    private float bookPrice;

    public String getBookName() {
        return bookName;
    }

    public void setBookName(String bookName) {
        this.bookName = bookName;
    }

    public float getBookPrice() {
        return bookPrice;
    }

    public void setBookPrice(float bookPrice) {
        this.bookPrice = bookPrice;
    }
}

(1)通过序列化在进程间传递对象【使用Parcelable】

顺带演示Serializable的使用方法。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <Button
        android:id="@+id/btn_intent"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="通过序列化在进程间传递对象"
        android:layout_margin="10dp"/>

    <Button
        android:id="@+id/btn_savefile"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="保存对象的字节序列到本地文件中"
        android:layout_margin="10dp"/>

    <Button
        android:id="@+id/btn_openfile"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="反序列化从文件中读取对象数据"
        android:layout_margin="10dp"/>
    <TextView
        android:id="@+id/tv_showbookInfo"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>
</LinearLayout>
activity_serialize_one.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="通过序列化Parcelable传递的对象数据"/>

    <TextView
        android:id="@+id/tv_Parcelable"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginBottom="10dp"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="通过序列化Serializable传递的对象数据"/>

    <TextView
        android:id="@+id/tv_Serializable"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

</LinearLayout>
activity_serialize_two.xml
package com.why.project.androidcnblogsdemo.activity;

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.TextView;

import com.why.project.androidcnblogsdemo.R;
import com.why.project.androidcnblogsdemo.bean.BookBean;
import com.why.project.androidcnblogsdemo.bean.UserBean;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

/**
 * Created by HaiyuKing
 * Used 序列化的首页界面
 */

public class SerializeOneActivity extends AppCompatActivity{

    private UserBean mUserBean;
    private BookBean mBookBean;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_serialize_one);

        initDatas();
        initEvents();

    }

    private void initDatas() {
        mUserBean = new UserBean();
        mUserBean.setUserId(1111);
        mUserBean.setUserName("haiyuKing");
        mUserBean.setVip(true);

        mBookBean = new BookBean();
        mBookBean.setBookName("Android开发艺术探索");
        mBookBean.setBookPrice(79f);
    }

    private void initEvents() {
        findViewById(R.id.btn_intent).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Intent intent = new Intent(SerializeOneActivity.this, SerializeTwoActivity.class);
                Bundle bundle = new Bundle();
                bundle.putParcelable("bundle_user",mUserBean);
                bundle.putSerializable("bundle_book",mBookBean);
                intent.putExtras(bundle);
                startActivity(intent);
            }
        });


        findViewById(R.id.btn_savefile).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                
            }
        });

        findViewById(R.id.btn_openfile).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                

            }
        });
    }

}

接收:

package com.why.project.androidcnblogsdemo.activity;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.TextView;

import com.why.project.androidcnblogsdemo.R;
import com.why.project.androidcnblogsdemo.bean.BookBean;
import com.why.project.androidcnblogsdemo.bean.UserBean;

/**
 * Created by HaiyuKing
 * Used 序列化的第二个界面
 */

public class SerializeTwoActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_serialize_two);

        TextView tv_Parcelable = findViewById(R.id.tv_Parcelable);
        TextView tv_Serializable = findViewById(R.id.tv_Serializable);

        UserBean userBean = (UserBean)getIntent().getParcelableExtra("bundle_user");
        BookBean bookBean = (BookBean)getIntent().getSerializableExtra("bundle_book");

        tv_Parcelable.setText(userBean.getUserId()+";"+userBean.getUserName()+";"+userBean.isVip());
        tv_Serializable.setText(bookBean.getBookName()+";"+bookBean.getBookPrice());
    }
}

效果图

(2)保存对象的字节序列到本地文件中【使用Serializable】

package com.why.project.androidcnblogsdemo.activity;

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.TextView;

import com.why.project.androidcnblogsdemo.R;
import com.why.project.androidcnblogsdemo.bean.BookBean;
import com.why.project.androidcnblogsdemo.bean.UserBean;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

/**
 * Created by HaiyuKing
 * Used 序列化的首页界面
 */

public class SerializeOneActivity extends AppCompatActivity{

    private UserBean mUserBean;
    private BookBean mBookBean;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_serialize_one);

        initDatas();
        initEvents();

    }

    private void initDatas() {
        mUserBean = new UserBean();
        mUserBean.setUserId(1111);
        mUserBean.setUserName("haiyuKing");
        mUserBean.setVip(true);

        mBookBean = new BookBean();
        mBookBean.setBookName("Android开发艺术探索");
        mBookBean.setBookPrice(79f);
    }

    private void initEvents() {
        findViewById(R.id.btn_intent).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
               
            }
        });


        findViewById(R.id.btn_savefile).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                try {
                    //文件存放的位置在 /data/data/<package>/files 下
                    File file = new File(SerializeOneActivity.this.getFilesDir().getPath(), "cache.txt");
                    ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(file));
                    out.writeObject(mBookBean);
                    out.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }

            }
        });

        findViewById(R.id.btn_openfile).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                try {
                    //文件存放的位置在 /data/data/<package>/files 下
                    File file = new File(SerializeOneActivity.this.getFilesDir().getPath(), "cache.txt");
                    ObjectInputStream out = new ObjectInputStream(new FileInputStream(file));
                    BookBean newBookBean = (BookBean) out.readObject();//内容一致,但不是同一个对象
                    out.close();

                    TextView tv_showbookInfo = findViewById(R.id.tv_showbookInfo);
                    tv_showbookInfo.setText(newBookBean.getBookName()+";"+newBookBean.getBookPrice());
                } catch (IOException e) {
                    e.printStackTrace();
                } catch (ClassNotFoundException e) {
                    e.printStackTrace();
                }

            }
        });
    }

}

 cache.txt的路径:

效果图:

(3)将对象序列化后通过网络传输【使用Serializable】

暂时空缺

参考资料

序列化Serializable和Parcelable的理解和区别

《Android开发艺术探索》

Android中Parcelable接口用法

Android 读写文件 FileNotFoundException 异常

原文地址:https://www.cnblogs.com/whycxb/p/9394413.html