Parcelable实战

Android开发过程中,无法将对象的引用传给Activities或者Fragments,我们需要将这些对象放到一个Intent或者Bundle里面,然后再传递,这就需要用到序列化和反序列化。

  • 序列化分类
  • Parcelable原理
  • Parcelable使用
  • 回调接口传递
  • 注意事项

序列化分类

(1)Serializable 是java的序列化技术,使用简单,频繁的IO操作,效率低

(2)Parcelable 是Android为我们提供的序列化的接口,使用稍微复杂,内存操作,效率高

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

Parcelable原理

(1)Parcelable是Android特有的序列化API,它的出现是为了解决Serializable在序列化的过程中消耗资源严重的问题,但是因为本身使用需要手动处理序列化和反序列化过程,会与具体的代码绑定,使用较为繁琐,一般只获取内存数据的时候使用。

(2)Parcelable依赖于Parcel,Parcel的意思是包装,实现原理是在内存中建立一块共享数据块,序列化和反序列化均是操作这一块的数据,如此来实现。


在这里插入图片描述

Parcelable使用

1、首先创建一个ParcelBean对象,继承自Parcelable,这个就是我们要将其序列化的对象,对象里边的属性有数值、boolean、字符串、对象、数组和链表等各种类型,重点是看下各种类型的序列化和反序列化的方式。

import android.os.Parcel;
import android.os.Parcelable;
import java.util.ArrayList;
import java.util.List;

public class ParcelBean implements Parcelable{

    public int count;
    public boolean flag;
    public String name;
    public ParcelItemBean itemBean;
    public List itmeList = new ArrayList<>();
    public String[] array;
    //回调接口对象
    public ParcelCallbackBean parcelCallbackBean;

    public ParcelBean() {
        super();
    }

    public ParcelBean(Parcel source) {
        super();
        count = source.readInt();
        flag = source.readByte() != 0;
        name = source.readString();
        itemBean= source.readParcelable(ParcelItemBean.class.getClassLoader()); // 读取对象需要提供一个类加载器去读取,因为写入的时候写入了类的相关信息
        source.readTypedList(itmeList, ParcelItemBean.CREATOR); //对应writeTypeList
        array = source.createStringArray();
        //回调接口
        parcelCallbackBean= source.readParcelable(ParcelCallbackBean.class.getClassLoader());
    }

    public static final Creator CREATOR = new Creator() {
        @Override
        public ParcelBean createFromParcel(Parcel in) {
            return new ParcelBean(in);
        }

        @Override
        public ParcelBean[] newArray(int size) {
            return new ParcelBean[size];
        }
    };

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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(count);
        dest.writeByte((byte) (flag ? 1 : 0));
        dest.writeString(name);
        dest.writeParcelable(itemBean, flags);
        dest.writeTypedList(itmeList);
        dest.writeStringArray(array);
        //回调接口对象
        dest.writeParcelable(parcelCallbackBean, flags);
    }
}

2、里边有一个ParcelItemBean的子对象,如下

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

public class ParcelItemBean implements Parcelable {
    public String itemName;

    public ParcelItemBean(){
        super();
    }

    protected ParcelItemBean(Parcel in) {
        super();
        itemName = in.readString();
    }

    public static final Creator CREATOR = new Creator() {
        @Override
        public ParcelItemBean createFromParcel(Parcel in) {
            return new ParcelItemBean(in);
        }

        @Override
        public ParcelItemBean[] newArray(int size) {
            return new ParcelItemBean[size];
        }
    };

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

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

3、对象序列化后需要通过Activity来加到ParcelableActivity的Intent中发送出去

import java.util.ArrayList;

public class ParcelableActivity extends AppCompatActivity implements View.OnClickListener {

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

        initView();
    }

    private void initView() {
        findViewById(R.id.parcel).setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.parcel:
                parcelSend();
                break;
            default:
                break;
        }
    }

    private void parcelSend() {
        ParcelBean parcelBean = new ParcelBean();
        parcelBean.name = "lisi";
        parcelBean.flag = false;
        parcelBean.count = 66;
        //写入对象
        ParcelItemBean parcelItemBean = new ParcelItemBean();
        parcelItemBean.itemName = "itmeNNN";
        parcelBean.itemBean = parcelItemBean;
        //写入list
        ParcelItemBean itemBean1 = new ParcelItemBean();
        itemBean1.itemName = "item111";
        ParcelItemBean itemBean2 = new ParcelItemBean();
        itemBean2.itemName = "item222";
        ArrayList beanList = new ArrayList<>();
        beanList.add(itemBean1);
        beanList.add(itemBean2);
        parcelBean.itmeList = beanList;
        //写入数组
        String array[] = new String[]{"num1", "num2", "num3"};
        parcelBean.array = array;
        //写入回调接口
        ParcelCallbackListenerBinder.CallbackListener callbackListener = new ParcelCallbackListenerBinder.CallbackListener() {
            @Override
            public void onComplete(Object obj) {
                Log.d("parcel_call_back", "callBackComplete");
            }

            @Override
            public void onError(String msg) {
                Log.d("parcel_call_back", "callBackError");
            }

            @Override
            public void onCancel() {
                Log.d("parcel_call_back", "callBackCancel");
            }
        };
        ParcelCallbackListenerBinder.ClickCallbackListener clickCallbackListener = new ParcelCallbackListenerBinder.ClickCallbackListener() {
            @Override
            public void onClick(String channel) {
                Log.d("parcel_call_back", "clickCallBack"+channel);
            }
        };
        ParcelCallbackBean parcelCallbackBean = new ParcelCallbackBean(new ParcelCallbackListenerBinder(callbackListener, clickCallbackListener));
        parcelBean.parcelCallbackBean = parcelCallbackBean;

        Intent intent = new Intent(this, ParcelReceiveActivity.class);
        intent.putExtra("keyName", "zhangsan");
        intent.putExtra("parcel", parcelBean);
        startActivity(intent);

    }
}

4、上面是发送的Activity,还需要一个接收的ParcelReceiveActivity


public class ParcelReceiveActivity extends AppCompatActivity implements View.OnClickListener {

    ParcelCallbackBean mParcelCallbackBean;

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

        initView();
    }

    private void initView() {
        findViewById(R.id.click_callback).setOnClickListener(this);
        Intent intent = getIntent();
        Log.d("ParcelReceiveActivity", "keyName = "+intent.getStringExtra("keyName"));

        ParcelBean parcelBean = intent.getParcelableExtra("parcel");
        Log.d("ParcelReceiveActivity", "parcelBean.name = "+parcelBean.name);
        Log.d("ParcelReceiveActivity", "parcelBean.flag = "+parcelBean.flag);
        Log.d("ParcelReceiveActivity", "parcelBean.count = "+parcelBean.count);
        Log.d("ParcelReceiveActivity", "parcelBean.itemBean.itemName = "+parcelBean.itemBean.itemName);
        Log.d("ParcelReceiveActivity", "parcelBean.itmeList.get(0).itemName = "+parcelBean.itmeList.get(0).itemName);
        Log.d("ParcelReceiveActivity", "parcelBean.itmeList.get(1).itemName = "+parcelBean.itmeList.get(1).itemName);
        Log.d("ParcelReceiveActivity", "parcelBean.array[0] = "+parcelBean.array[0]);
        Log.d("ParcelReceiveActivity", "parcelBean.array[1] = "+parcelBean.array[1]);

        //获取传递过来的接口对象
        mParcelCallbackBean = parcelBean.parcelCallbackBean;


    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.click_callback:
                clickCallBack();
                break;
            default:
                break;
        }
    }

    private void clickCallBack() {
        mParcelCallbackBean.parcelCallbackListenerBinder.getClickCallbackListener().onClick("我是传递过来的点击事件");
    }
}

5、在接收的Activity里边打印Log,如下

    ParcelReceiveActivity: keyName = zhangsan
    parcelBean.name = lisi
    parcelBean.flag = false
    parcelBean.count = 66
    parcelBean.itemBean.itemName = itmeNNN
    parcelBean.itmeList.get(0).itemName = item111
    parcelBean.itmeList.get(1).itemName = item222
    parcelBean.array[0] = num1
    parcelBean.array[1] = num2

回调接口传递

上面有一个回调接口的对象的传递,这里单独拿出来说一下,首先建立一个ParcelCallbackListenerBinder对象,继承自Binder,用来管理这两个回调接口

import android.os.Binder;

public class ParcelCallbackListenerBinder extends Binder{
    public CallbackListener callbackListener;
    public ClickCallbackListener clickCallbackListener;

    public ParcelCallbackListenerBinder(CallbackListener callbackListener, ClickCallbackListener clickCallbackListener) {
        this.callbackListener = callbackListener;
        this.clickCallbackListener = clickCallbackListener;
    }

    public CallbackListener getCallbackListener() {
        return callbackListener;
    }

    public ClickCallbackListener getClickCallbackListener() {
        return clickCallbackListener;
    }


    public static interface CallbackListener {
        void onComplete(Object obj);

        void onError(String msg);

        void onCancel();
    }

    public static interface ClickCallbackListener {
        void onClick(String channel);
    }
}

然后再将这个封装的回调接口,封装到ParcelCallbackBean中,这个也是继承自Parcelable的

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

import com.jd.test.phototest.activity.interfac.ParcelCallbackListenerBinder;

public class ParcelCallbackBean implements Parcelable{

    public ParcelCallbackListenerBinder parcelCallbackListenerBinder;

    public ParcelCallbackBean(ParcelCallbackListenerBinder binder) {
        parcelCallbackListenerBinder = binder;
    }

    protected ParcelCallbackBean(Parcel source) {
        Object object = source.readValue(ParcelCallbackListenerBinder.class.getClassLoader());
        if (object instanceof ParcelCallbackListenerBinder) {
            parcelCallbackListenerBinder = (ParcelCallbackListenerBinder) object;
        }
    }

    public static final Creator CREATOR = new Creator() {
        @Override
        public ParcelCallbackBean createFromParcel(Parcel in) {
            return new ParcelCallbackBean(in);
        }

        @Override
        public ParcelCallbackBean[] newArray(int size) {
            return new ParcelCallbackBean[size];
        }
    };

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

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

在ParcelReceiveActivity里进行回调,通过clickCallBack方法进行回调,在ParcelableActivity里进行回调的接收。我们来看一下点击回调的结果

parcel_call_back: clickCallBack我是传递过来的点击事件
parcel_call_back: clickCallBack我是传递过来的点击事件
parcel_call_back: clickCallBack我是传递过来的点击事件

注意事项

(1)Parcelable使用的时候要注意ParcelBean和writeToParcel也就是读写时各个属性的顺序,一定要一致,不然会存在问题。

(2)接口回调的ParcelCallbackListenerBinder类要继承自Binder,这个问啥我还没搞清楚,猜测应该和Parcelable底层使用的Binder进程通信有关,欢迎各位大佬指教。

尊重作者,尊重原创,参考文章:
https://www.jianshu.com/p/df35baa91541
https://blog.csdn.net/hacker_crazy/article/details/80840868
https://www.cnblogs.com/tangZH/p/10998065.html

你可能感兴趣的:(Parcelable实战)