序列化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 。
使用方法
新建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 读写文件 FileNotFoundException 异常
序列化Serializable和Parcelable的更多相关文章
- 转 序列化Serializable和Parcelable的区别详解
什么是序列化,为什么要进行序列化 答:对象要进行传输(如:activity 与activity间 ,网络间 进程间等等).存储到本地就必须进行序列化 . 这种可传输的状态就是序列化. 怎么序列化??两 ...
- Android序列化之Serializable和Parcelable
PS:还有几天就开学了.先来一发. 学习内容: 1.序列化的目的 2.Android中序列化的两种方式 3.Parcelable与Serializable的性能比较 4.Android中如何使用Par ...
- Android中Serializable和Parcelable序列化对象详解
学习内容: 1.序列化的目的 2.Android中序列化的两种方式 3.Parcelable与Serializable的性能比较 4.Android中如何使用Parcelable进行序列化操作 5.P ...
- Serializable 和 Parcelable 区别
http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2015/0204/2410.html http://www.cnblogs.com/blu ...
- 浅谈Android中Serializable和Parcelable使用区别
版权声明:本文出自汪磊的博客,转载请务必注明出处. 一.概述 Android开发的时候,我们时长遇到传递对象的需求,但是我们无法将对象的引用传给Activity或者Fragment,我们需要将这些对象 ...
- Serializable 和 parcelable的实现和比较
首先这个两个接口都是用来序列化对象的 但是两者在性能和应用场合上有区别,parcelable的性能更好,但是在需要保存或者网络传输的时候需要选择Serializable因为parcelable版本在不 ...
- Serializable和Parcelable的简单介绍
Serializable和Pacelable接口可以完成对象的序列化过程,当我们需要通过Intent和Binder传输数据时就需要使用Parcelable或者Serializable. Seriali ...
- Intent传递对象——Serializable和Parcelable差别
前两篇文章讨论了Serializable和Parcelable实现Intent之间传递对象和对象数组的方式.两种方法实现上相似,效果一致,怎么选择用哪种方法实现呢? Intent在不同的组件中传递对象 ...
- Serializable 和Parcelable 详解
序列化:为了保存在内存中的各种对象的状态,并可以把保存的对象的状态读出来 安卓中实现序列化的接口有两个,一个是serializable,一个是parcelable. 一.实现序列化: 1.是可以将对象 ...
随机推荐
- python 获取mac地址zz
通过python获取当前mac地址的方法如下:(1)通用方法,借助uuid模块def get_mac_address(): import uuid node = uuid.getnode() ...
- anjular分页组件tm-pagination的使用
原组件地址:https://github.com/miaoyaoyao/AngularJs-UI (1)直接从git上clone下来的demo无法正常显示,后来重新到在线的demo上拷贝了templa ...
- golang实现障碍、转弯最少的A*寻路
目录 目标: 要点: 源码: 目标: 优先寻找无障碍的路径 目标不可达时,寻找障碍最少的路径 路径长度相等时,优先转弯最少的路径 多个目标点时,根据以上要求到达其中一个目标点即可 要点: 最优格子的选 ...
- [POJ1961]Period (KMP)
题意 求字符串s的最小循环元长度和循环次数 思路 s[1~i]满足循环元要len能整除i并且s[len+1~i]=s[1~i-len] 代码 #include<cstdio> #inclu ...
- Chrome_高亮显示当前改变的区域
- vue变异方法
push() 往数组最后面添加一个元素,成功返回当前数组的长度 pop() 删除数组的最后一个元素,成功返回删除元素的值 shift() 删除数组的第一个元素,成功返回删除元素的值u ...
- Linux下不同文件不同颜色的意义
通常,我们使用ls查看文件时,会出现不同颜色的文件,今天我就大概写一下不同颜色的代表意义: 蓝色————目录: 绿色————可执行文件: 红色————压缩文件: 浅蓝色————链接文件: 紫红色——— ...
- swust oj 1010
折半查找的实现 1000(ms) 10000(kb) 2877 / 11213 编写程序实现折半查找算法. 输入 第一行是查找表的长度n 第二行是查找表中的数据元素 : 第三行是要查找的数据元素的关键 ...
- XLua----热更新
一.xLua 环境配置 1).Xlua中 Plugin Xlua复制到 需要热更新的工程中---->Assets子目录 2).开启宏HOTFIX_ENABLE File---->bui ...
- muse-ui底部导航自定义图标和字体颜色
最近在鼓捣用vue.js进行混合APP开发,遍寻许久终于找到muse-ui这款支持vue的轻量级UI框架,竟还支持按需引入,甚合萝卜意! 底部导航的点击波纹特效也是让我无比惊喜,然而自定义图标和字体颜 ...