上一篇中我们透过源码看到了Parcel背后的机制,本质上把它当成一个Serialize就可以了,只是它是在内存中完成的序列化和反序列化,利用的是连续的内存空间,因此会更加高效。
我们接下来要说的是Parcel类如何应用。就应用程序而言,最常见使用Parcel类的场景就是在Activity间传递数据。没错,在Activity间使用Intent传递数据的时候,可以通过Parcelable机制传递复杂的对象。
在下面的程序中,MyColor用于保存一个颜色值,MainActivity在用户点击屏幕时将MyColor对象设成红色,传递到SubActivity中,此时SubActivity的TextView显示为红色的背景;当点击SubActivity时,将颜色值改为绿色,返回MainActivity,期望的是MainActivity的TextView显示绿色背景。
来看一下MyColor类的实现代码:
- package com.wenbin.test;
- import android.graphics.Color;
- import android.os.Parcel;
- import android.os.Parcelable;
- /**
- * @author 曹文斌
- * http://blog.csdn.net/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的代码:
- 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.csdn.net/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的代码:
- 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.csdn.net/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的代码:
- <?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,代码如下:
- package com.wenbin.test;
- import android.os.Binder;
- /**
- * @author 曹文斌
- * http://blog.csdn.net/caowenbin
- *
- */
- public class BinderData extends Binder {
- public int color;
- }
修改MyColor的代码如下:
- package com.wenbin.test;
- import android.graphics.Color;
- import android.os.Parcel;
- import android.os.Parcelable;
- /**
- * @author 曹文斌
- * http://blog.csdn.net/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")一句,变成:
- @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的开发文档了。
相关推荐
在Android开发中,数据的序列化是一个非常重要的概念,它涉及到数据的持久化存储和跨进程通信。序列化是将对象的状态转化为可存储或可传输的形式的过程,而在Android中,我们通常会用到两种主要的序列化方式:...
在标题"android 对象序列化"中,我们要讨论的是如何在Android环境中实现对象的序列化,并特别是在通过Intent传递对象的应用场景。 对象序列化在Android中的应用主要包括以下几个方面: 1. 数据持久化:当应用程序...
在Android开发中,对象序列化是一项重要的技术,它允许我们将对象的状态转化为可存储或传输的格式,例如保存到文件、数据库或通过网络进行传递。Android提供了两种主要的对象序列化方式:Parcelable和Serializable。...
在Android开发中,序列化(Serialization)是一种将对象转换为可传输或存储格式的过程,而SharedPreferences则是一个轻量级的数据存储机制,常用于保存应用程序的简单配置数据。本篇文章将详细探讨如何将Android序列...
3. 使用难度:Serializable接口只需简单实现,而Parcelable需要手动编写序列化和反序列化的代码,维护成本较高。 4. 应用场景:Serializable适用于简单的序列化需求,如保存用户偏好设置;Parcelable更适合需要高效...
三、常见的JSON反序列化库 1. Java:Jackson、Gson、org.json等 2. Python:json模块、ujson、simplejson等 3. JavaScript:JSON.parse() 内置方法 4. .NET:Newtonsoft.Json(Json.NET) 5. PHP:json_decode 函数 ...
`Parcelable`是Android提供的一种高效的数据序列化方式,相比`Serializable`,它的序列化和反序列化速度更快,但实现过程相对复杂。 标题"Android Parcelable序列化自定义类集合在Activity间传递"所涉及的知识点...
在Android开发中,与Web服务交互是常见的需求,这通常涉及到对象的序列化和反序列化过程。Web服务,特别是基于SOAP(Simple Object Access Protocol)的,需要将数据转换为可传输的格式,如XML,以便在客户端和...
Parcelable是Android特有的序列化方式,相比Serializable,它的性能更好,但是实现较为复杂。你需要手动编写序列化和反序列化的代码,使用`writeToParcel()`和`Creator`接口。 优点: - 性能高效,Parcelable比...
序列化是将对象转换为可存储或传输格式的过程,而在Android中,有两种主要的序列化方式:Parcelable和Serializable。下面将详细介绍这两种序列化机制及其在数据传递中的应用。 **Parcelable** 是Android特有的序列...
Android并没有内置的XML序列化库,但可以借助第三方库如XStream或Jackson,或者手动实现序列化。手动实现通常涉及创建`toString()`方法,将对象属性转化为XML格式的字符串。 2. **Gson**: 虽然Gson主要用于JSON序列...
在Android开发中,数据序列化是一个非常重要的概念,它允许对象的状态被保存和恢复,以便在不同的时间点或不同的环境中重建对象。Parcelable是Android平台提供的一种高效的数据序列化方式,比传统的Serializable接口...
在Android系统中关于序列化的方法一般有两种,分别是实现Serializable接口和Parcelable接口,其中Serializable接口是来自Java中的序列化接口,而Parcelable是Android自带的序列化接口。 上述的两种序列化接口都有各自...
本教程将详细讲解如何在C#中实现`TreeView`控件的序列化和反序列化,这对于数据持久化和用户界面状态的保存至关重要。 首先,让我们了解序列化和反序列化的基本概念。**序列化**是将对象的状态转换为可存储或传输的...
### IDEA中实体类实现序列化接口与序列化ID生成 #### 一、序言 在Java开发中,序列化是一项非常重要的技术,它允许将对象的状态转换为字节流,从而方便在网络上传输或者存储到磁盘上。为了保证序列化的兼容性与...
在Android平台上,Java对象反序列化是一个常见的操作,它涉及到将序列化的数据转换回原来的对象实例。这在数据存储、网络通信以及跨进程通信(IPC)等场景中扮演着重要角色。本文将深入探讨Android上Java对象反序列...
除了使用第三方库,Android还提供了一种基于注解的序列化方式,即`org.w3c.dom.Element`和`javax.xml.transform`。这种方式相对复杂,但可以避免引入额外的依赖。你可以创建一个继承自`Serializable`的类,然后使用`...
3. 尝试反序列化之前序列化的对象,观察因版本不匹配导致的异常。 4. 解释如何为类显式声明`serialVersionUID`,通常在类中添加`private static final long serialVersionUID = 1L;`,并讨论为何这样做可以避免版本...
为了实现对象序列化,我们通常会使用Java内置的序列化机制,或者第三方库如Gson、Jackson或Protobuf。 1. **Java内置序列化**:Java提供了一个标准的序列化接口`java.io.Serializable`,只需让要序列化的类实现这个...
**三、序列化与反序列化的细节** 1. 复杂类型的处理:除了基本类型外,还可以序列化和反序列化自定义类、结构体以及容器(如vector、map等)。对于自定义类型,通常需要重载`operator和`operator>>`,或者使用...