Android的Parcel机制

转载至:http://blog.youkuaiyun.com/caowenbin/article/details/6532238

上一篇中我们透过源码看到了Parcel背后的机制,本质上把它当成一个Serialize就可以了,只是它是在内存中完成的序列化和反序列化,利用的是连续的内存空间,因此会更加高效。
我们接下来要说的是Parcel类如何应用。就应用程序而言,最常见使用Parcel类的场景就是在Activity间传递数据。没错,在Activity间使用Intent传递数据的时候,可以通过Parcelable机制传递复杂的对象。
在下面的程序中,MyColor用于保存一个颜色值,MainActivity在用户点击屏幕时将MyColor对象设成红色,传递到SubActivity中,此时SubActivity的TextView显示为红色的背景;当点击SubActivity时,将颜色值改为绿色,返回MainActivity,期望的是MainActivity的TextView显示绿色背景。
来看一下MyColor类的实现代码:

view plain
package com.wenbin.test;

import android.graphics.Color;
import android.os.Parcel;
import android.os.Parcelable;

/**
* @author 曹文斌
* http://blog.youkuaiyun.com/caowenbin
*
*/
public class MyColor implements Parcelable {
private int color=Color.BLACK;

MyColor(){
color=Color.BLACK;
}

MyColor(Parcel in){
color=in.readInt();
}

public int getColor(){
return color;
}

public void setColor(int color){
this.color=color;
}

@Override
public int describeContents() {
return 0;
}

@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(color);
}

public static final Parcelable.Creator<MyColor> CREATOR
= new Parcelable.Creator<MyColor>() {
public MyColor createFromParcel(Parcel in) {
return new MyColor(in);
}

public MyColor[] newArray(int size) {
return new MyColor[size];
}
};
}


该类实现了Parcelable接口,提供了默认的构造函数,同时也提供了可从Parcel对象开始的构造函数,另外还实现了一个static的构造器用于构造对象和数组。代码很简单,不一一解释了。
再看MainActivity的代码:

view plain
package com.wenbin.test;

import android.app.Activity;
import android.content.Intent;
import android.graphics.Color;
import android.os.Bundle;
import android.view.MotionEvent;

/**
* @author 曹文斌
* http://blog.youkuaiyun.com/caowenbin
*
*/
public class MainActivity extends Activity {
private final int SUB_ACTIVITY=0;
private MyColor color=new MyColor();

@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
}

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode==SUB_ACTIVITY){
if (resultCode==RESULT_OK){
if (data.hasExtra("MyColor")){
color=data.getParcelableExtra("MyColor"); //Notice
findViewById(R.id.text).setBackgroundColor(color.getColor());
}
}
}
}

@Override
public boolean onTouchEvent(MotionEvent event){
if (event.getAction()==MotionEvent.ACTION_UP){
Intent intent=new Intent();
intent.setClass(this, SubActivity.class);
color.setColor(Color.RED);
intent.putExtra("MyColor", color);
startActivityForResult(intent,SUB_ACTIVITY);
}
return super.onTouchEvent(event);
}

}


下面是SubActivity的代码:

view plain
package com.wenbin.test;

import android.app.Activity;
import android.content.Intent;
import android.graphics.Color;
import android.os.Bundle;
import android.view.MotionEvent;
import android.widget.TextView;

/**
* @author 曹文斌
* http://blog.youkuaiyun.com/caowenbin
*
*/
public class SubActivity extends Activity {
private MyColor color;

@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
((TextView)findViewById(R.id.text)).setText("SubActivity");
Intent intent=getIntent();
if (intent!=null){
if (intent.hasExtra("MyColor")){
color=intent.getParcelableExtra("MyColor");
findViewById(R.id.text).setBackgroundColor(color.getColor());
}
}
}

@Override
public boolean onTouchEvent(MotionEvent event){
if (event.getAction()==MotionEvent.ACTION_UP){
Intent intent=new Intent();
if (color!=null){
color.setColor(Color.GREEN);
intent.putExtra("MyColor", color);
}
setResult(RESULT_OK,intent);
finish();
}
return super.onTouchEvent(event);
}
}


下面是main.xml的代码:

view plain
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/hello"
android:id="@+id/text"
/>
</LinearLayout>


注意的是在MainActivity的onActivityResult()中,有一句color=data.getParcelableExtra("MyColor"),这说明的是反序列化后是一个新的MyColor对象,因此要想使用这个对象,我们做了这个赋值语句。
记得在上一篇《探索Android中的Parcel机制(上)》中提到,如果数据本身是IBinder类型,那么反序列化的结果就是原对象,而不是新建的对象,很显然,如果是这样的话,在反序列化后在MainActivity中就不再需要color=data.getParcelableExtra("MyColor")这句了。因此,换一种MyColor的实现方法,令其中的int color成员变量使用IBinder类型的成员变量来表示。
新建一个BinderData类继承自Binder,代码如下:

view plain
package com.wenbin.test;

import android.os.Binder;

/**
* @author 曹文斌
* http://blog.youkuaiyun.com/caowenbin
*
*/
public class BinderData extends Binder {
public int color;
}


修改MyColor的代码如下:

view plain
package com.wenbin.test;

import android.graphics.Color;
import android.os.Parcel;
import android.os.Parcelable;

/**
* @author 曹文斌
* http://blog.youkuaiyun.com/caowenbin
*
*/
public class MyColor implements Parcelable {
private BinderData data=new BinderData();

MyColor(){
data.color=Color.BLACK;
}

MyColor(Parcel in){
data=(BinderData) in.readValue(BinderData.class.getClassLoader());
}

public int getColor(){
return data.color;
}

public void setColor(int color){
data.color=color;
}

@Override
public int describeContents() {
return 0;
}

@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeValue(data);
}

public static final Parcelable.Creator<MyColor> CREATOR
= new Parcelable.Creator<MyColor>() {
public MyColor createFromParcel(Parcel in) {
return new MyColor(in);
}

public MyColor[] newArray(int size) {
return new MyColor[size];
}
};
}

去掉MainActivity的onActivityResult()中的color=data.getParcelableExtra("MyColor")一句,变成:

view plain
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode==SUB_ACTIVITY){
if (resultCode==RESULT_OK){
if (data.hasExtra("MyColor")){
findViewById(R.id.text).setBackgroundColor(color.getColor());
}
}
}
}

再次运行程序,结果符合预期。

以上就是Parcel在应用程序中的使用方法,与Serialize还是挺相似的,详细的资料当然还是要参考Android SDK的开发文档了。
### AndroidParcel 的使用方法 #### 序列化与反序列化的机制 `Parcel` 是一种高效的序列化工具,在 `Android` 平台被广泛应用于进程间通信 (IPC)[^2]。为了使自定义类能够支持这种序列化方式,开发者需要让其继承并实现 `Parcelable` 接口。 #### 实现 Parcelable 接口 要创建可序列化的对象,需遵循如下模式: 1. **声明 CREATOR 字段** 类中应包含静态字段 `CREATOR`,它实现了 `Parcelable.Creator<T>` 接口,负责实例化新对象。 2. **重写 write.Parcel 方法** 将对象的数据成员按顺序写入到 `Parcel` 对象里。 3. **提供公共构造函数读取 parcel 数据** 构造器接收一个 `Parcel` 参数来初始化当前对象的状态。 ```java public class MyObject implements Parcelable { private String name; private int age; public static final Creator<MyObject> CREATOR = new Creator<MyObject>() { @Override public MyObject createFromParcel(Parcel source) { return new MyObject(source); } @Override public MyObject[] newArray(int size) { return new MyObject[size]; } }; protected MyObject(Parcel in) { this.name = in.readString(); this.age = in.readInt(); } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeString(name); dest.writeInt(age); } @Override public int describeContents() { return 0; } } ``` #### 使用 Intent 或 Bundle 进行传输 一旦实现了上述逻辑,则可以通过 `Intent` 或者 `Bundle` 来携带此类对象: ```java // 创建一个新的MyObject实例 MyObject obj = new MyObject(); // 放置在Intent中 intent.putExtra("key", obj); // 获取传来的obj MyObject receivedObj = intent.getExtras().getParcelable("key"); ``` #### 注意事项 对于不同版本的 Android 系统,`Parcel` 内部的对象池管理有所差异;特别是从 Android 31 开始引入了一些新的变化[^1]。另外需要注意的是,在某些特定情况下(比如跨平台服务),可能会遇到字节对齐的问题,这可能导致数据丢失或损坏的情况发生[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值