Android开发笔记(二十七)对象序列化

来源:互联网 发布:软件项目监理费用 编辑:程序博客网 时间:2024/05/16 09:35

什么是序列化

程序中存储和传递信息,需要有个合适的数据结构,最简单的是定义几个变量,变量多了之后再分门别类,便成了聚合若干变量的对象。代码在函数调用时可以直接传递对象,但更多的场合例如与文件交互、与网络交互、组件之间交互等等,就无法直接使用未经处理的对象。因此Java引入了序列化的概念,用于把一个对象转换为字节序列,然后再对这个字节序列做存储和传递操作。与之对应的是反序列化,反序列化是把一个字节序列恢复为Java对象的过程,而序列化是把Java对象转化为字节序列的过程。


Serializable

Serializable是Java设计用来定义序列化的接口,一个希望进行序列化的对象,需要实现Serializable接口。像上节《Android开发笔记(二十六)Java的容器类》中介绍的Java容器类,从队列到映射,其实在各自的基类容器之外,也都实现了Serializable接口。也就是说,这些Java容器类其实都是可序列化的对象。另外,我们常见的变量类型如String、Integer等等,也实现了Serializable接口。既然这些对象都是可序列化的,那就可以把对象用IO写到文件里,之后再可以从文件里读出原对象,读出的变量值与之前的变量值是一样的。


实现简单的Serializable接口无需自己重写任何序列化函数,只要提供一个序列化版本的id(serialVersionUID),Java便会对这个对象进行高效的序列化操作。但由于Serializable方式使用了反射机制,使得序列化的过程相对较慢。并且,这种机制会在序列化的时候创建许多的临时对象,容易触发垃圾回收。


需要注意的是,Serializable序列化不保存静态变量,另外使用Transient关键字可声明对指定字段不做序列化。对于某些复杂的对象,也可以重写writeObject、readObject方法来自定义序列化过程,比如队列、映射这些容器类就重写了writeObject和readObject方法。


Parcelable

Parcelable的设计初衷,便是因为Serializable方式较耗资源且执行速度偏慢,为此Android设计了Parcelable用于组件之间的消息传递(包括线程间传递与进程间传递)。Parcelable数据仅在内存中存在,所以在内存间数据传输时推荐使用;而Serializable可将数据持久化方便保存,所以在需要保存文件或网络传输数据时应选择Serializable。


采用Parcelable方式的类,需要自己定义如何打包(写数据)和解包(读数据),其余的序列化操作则由底层实现。具体需要实现的方法如下:
writeToParcel(Parcel out, int flags) : 写数据
Parcelable.Creator<类名> CREATOR : 例行公事实现createFromParcel(读数据)和newArray
describeContents : 返回0即可


序列化对象的消息传递

下面是Serializable和Parcelable两种方式在Activity之间传递消息的代码例子。
Serializable对象代码
import java.io.Serializable;public class SerData implements Serializable {    private static final long serialVersionUID = 999794470754667710L;    public int mAge;    public String mName;    public boolean mMarried;    public double mHeight;    }


Parcelable对象代码
import android.os.Parcel;import android.os.Parcelable;public class ParData implements Parcelable {public int mAge;public String mName;public boolean mMarried;public double mHeight;    // 写数据    @Override    public void writeToParcel(Parcel out, int flags) {        out.writeInt(mAge);        out.writeString(mName);        out.writeByte((byte) (mMarried ? 1 : 0));        out.writeDouble(mHeight);    }    // 例行公事实现createFromParcel和newArray    public static final Parcelable.Creator<ParData> CREATOR            = new Parcelable.Creator<ParData>() {        // 读数据        public ParData createFromParcel(Parcel in) {        ParData par = new ParData();        par.mAge = in.readInt();        par.mName = in.readString();        par.mMarried = in.readByte() != 0;        par.mHeight = in.readDouble();            return par;        }        public ParData[] newArray(int size) {            return new ParData[size];        }    };    @Overridepublic int describeContents() {return 0;}}

上面可以看到Parcelable暂不支持直接操作布尔boolean类型,但能间接通过设置byte位来实现boolean类型的参数传递。



主页面代码
import com.example.exmparcelable.data.ParData;import com.example.exmparcelable.data.SerData;import android.app.Activity;import android.content.Context;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.view.View.OnClickListener;import android.widget.AdapterView;import android.widget.AdapterView.OnItemSelectedListener;import android.widget.ArrayAdapter;import android.widget.Button;import android.widget.EditText;import android.widget.Spinner;import android.widget.Toast;public class MainActivity extends Activity implements OnClickListener {private EditText et_name, et_age, et_height;private Spinner sp_married;private String[] marriedStrArray = new String[]{"已婚", "未婚"};private boolean[] marriedBoolArray = new boolean[]{true, false};private boolean isMarried = true;private void setMarriedSpinner(Context context, int spinner_id, int seq) {sp_married = (Spinner) findViewById(spinner_id);ArrayAdapter<String> county_adapter;county_adapter = new ArrayAdapter<String>(context, R.layout.spinner_item, marriedStrArray);county_adapter.setDropDownViewResource(R.layout.spinner_dropdown_item);// setPrompt是设置弹出对话框的标题sp_married.setPrompt("请选择婚否");sp_married.setAdapter(county_adapter);sp_married.setOnItemSelectedListener(new SpinnerSelectedListener());if (seq >= 0) {sp_married.setSelection(seq, true);} else {sp_married.setFocusable(false);}}class SpinnerSelectedListener implements OnItemSelectedListener {public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) {isMarried = marriedBoolArray[arg2];}public void onNothingSelected(AdapterView<?> arg0) {}}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);et_name = (EditText) findViewById(R.id.et_name);et_age = (EditText) findViewById(R.id.et_age);et_height = (EditText) findViewById(R.id.et_height);setMarriedSpinner(this, R.id.sp_married, 0);Button btn_ser = (Button) findViewById(R.id.btn_ser);Button btn_par = (Button) findViewById(R.id.btn_par);btn_ser.setOnClickListener(this);btn_par.setOnClickListener(this);}@Overridepublic void onClick(View v) {if (et_age.getText()==null || et_age.getText().length()<=0) {Toast.makeText(this, "请输入年龄", Toast.LENGTH_LONG).show();return;} else if (et_name.getText()==null || et_name.getText().length()<=0) {Toast.makeText(this, "请输入姓名", Toast.LENGTH_LONG).show();return;} else if (et_height.getText()==null || et_height.getText().length()<=0) {Toast.makeText(this, "请输入身高", Toast.LENGTH_LONG).show();return;}if (v.getId() == R.id.btn_ser) {SerData ser = new SerData();ser.mAge = Integer.parseInt(et_age.getText().toString());ser.mName = et_name.getText().toString();ser.mMarried = isMarried;ser.mHeight = Double.parseDouble(et_height.getText().toString());Intent intent = new Intent(this, SerializableActivity.class);Bundle bundle = new Bundle();bundle.putSerializable("ser", ser);intent.putExtras(bundle);startActivity(intent);} else if (v.getId() == R.id.btn_par) {ParData par = new ParData();par.mAge = Integer.parseInt(et_age.getText().toString());par.mName = et_name.getText().toString();par.mMarried = isMarried;par.mHeight = Double.parseDouble(et_height.getText().toString());Intent intent = new Intent(this, ParcelableActivity.class);Bundle bundle = new Bundle();bundle.putParcelable("par", par);intent.putExtras(bundle);startActivity(intent);}}}


Serializable页面代码
import com.example.exmparcelable.data.SerData;import android.app.Activity;import android.os.Bundle;import android.view.Window;import android.widget.TextView;public class SerializableActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_ser);Bundle bundle = getIntent().getExtras();SerData ser = (SerData) bundle.getSerializable("ser");String desc = String.format("您输入的人物信息是:姓名%s,年龄%d,身高%f,婚否%b", ser.mName, ser.mAge, ser.mHeight, ser.mMarried);TextView tv_ser = (TextView) findViewById(R.id.tv_ser);tv_ser.setText(desc);}}


Parcelable页面代码
import com.example.exmparcelable.data.ParData;import android.app.Activity;import android.os.Bundle;import android.view.Window;import android.widget.TextView;public class ParcelableActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);setContentView(R.layout.activity_par);Bundle bundle = getIntent().getExtras();ParData par = bundle.getParcelable("par");String desc = String.format("您输入的人物信息是:姓名%s,年龄%d,身高%f,婚否%b", par.mName, par.mAge, par.mHeight, par.mMarried);TextView tv_par = (TextView) findViewById(R.id.tv_par);tv_par.setText(desc);}}



点击下载本文用到的对象序列化的工程代码



点此查看Android开发笔记的完整目录

0 0
原创粉丝点击