美文网首页
Parcel 解析 Bundle 数据

Parcel 解析 Bundle 数据

作者: gczxbb | 来源:发表于2021-01-08 00:03 被阅读0次

一、构建 Bundle

从 Parcel 中解析 出 Bundle 数据。
Parce readBundle() 方法。

public final Bundle readBundle(ClassLoader loader) {
    int length = readInt();
    if (length < 0) {
        return null;
    }

    final Bundle bundle = new Bundle(this, length);
    if (loader != null) {
        bundle.setClassLoader(loader);
    }
    return bundle;
}

创建 Bundle 对象
BaseBundle 类构造方法,根据数据长度,创建 ArrayMap 对象。

BaseBundle(Parcel parcelledData, int length) {
    readFromParcelInner(parcelledData, length);
}

将数据源 Parcel 入参

private void readFromParcelInner(Parcel parcel, int length) {
    if (length == 0) {
        mParcelledData = EMPTY_PARCEL;
        return;
    }
    ...
    Parcel p = Parcel.obtain();
    p.setDataPosition(0);
    p.appendFrom(parcel, offset, length);
    ...
    p.setDataPosition(0);
    mParcelledData = p;
}

从 Parcel 池 获取一个 Parcel,appendFrom() 入参数据源。内部赋值 mParcelledData。

构建 Bundle,并未立即解析 Parcel 数据 到 ArrayMap。

二、unparcel() 解析

unparcel() 方法,将 BaseBundle 内部 Parcel ,即 mParcelledData ,解析到 ArrayMap,在 Parcel 数据读取时用到。

synchronized void unparcel() {
    synchronized (this) {
        final Parcel parcelledData = mParcelledData;
        //Bundle内部parcelledData是空时,不用解析
        if (parcelledData == null) {
            return;
        }
        //Bundle内部parcelledData无数据时,不用解析
        if (isEmptyParcel()) {
            if (mMap == null) {
                mMap = new ArrayMap<>(1);
            } else {
                mMap.erase();
            }
            mParcelledData = null;
            return;
        }
        int N = parcelledData.readInt();
        ArrayMap<String, Object> map = mMap;
        //Map是空时创建。
        try {
            parcelledData.readArrayMapInternal(map, N, mClassLoader);
        } catch (BadParcelableException e) {
        } finally {
            mMap = map;
            parcelledData.recycle();
            mParcelledData = null;
        }
    }
}

Bundle 内部 Parcel,readArrayMapInternal() 方法。
解析完成,finally{} 释放 Parcel ,置空,因此,Intent 多次调用 getXxx() 时,在判空后不会再次 unparcel(),直接使用 ArrayMap。

void readArrayMapInternal(ArrayMap outVal, int N,
        ClassLoader loader) {
    int startPos;
    while (N > 0) {
        String key = readString();//从Parcel读取key
        Object value = readValue(loader);//从Parcel读取value
        outVal.append(key, value);//写入ArrayMap 
        N--;
    }
    outVal.validate();
}

遍历数量N,从 Parcel 中读取 key-value,写入 ArrayMap。
Parcel readValue() 方法

public final Object readValue(ClassLoader loader) {
        int type = readInt();

        switch (type) {
        case VAL_NULL:
            return null;

        case VAL_STRING:
            return readString();

        case VAL_INTEGER:
            return readInt();

        case VAL_PARCELABLE:
            return readParcelable(loader);

        ...
}

先读取 int 字段,JNI # nativeReadInt(),类型判断。
根据类型,调用 Parcel readXxx() 方法,JNI nativeReadXxx()方法。

Parcel readParcelable() 方法。

public final <T extends Parcelable> T readParcelable(ClassLoader loader) {
    Parcelable.Creator<?> creator = readParcelableCreator(loader);
    if (creator == null) {
        return null;
    }
    if (creator instanceof Parcelable.ClassLoaderCreator<?>) {
        Parcelable.ClassLoaderCreator<?> classLoaderCreator =
                (Parcelable.ClassLoaderCreator<?>) creator;
        return (T) classLoaderCreator.createFromParcel(this, loader);
    }
    return (T) creator.createFromParcel(this);
}

根据 ClassLoader ,查找 XxxBean 实现的 Creator。
调用 createFromParcel() 方法,创建 XxxBean 实体,传入 Parcel,读取字段向实体赋值。
(和写入数据的 XxxBean 实体非同一对象)

通过 unparcel() 方法,将 Parcel 中各种类型数据,全部解析 ArrayMap 数据结构。


任重到道远

相关文章

  • Parcel 解析 Bundle 数据

    一、构建 Bundle 从 Parcel 中解析 出 Bundle 数据。Parce readBundle() 方...

  • IPC—Messenger

    1.Messenger进程间通信时,bundle有Parcelable对象解析数据的时候,需要调用bundle.s...

  • Bundle Parcel 和 ClassLoader

    前言 前几天看到同事在讨论 动态代理 和 类加载器 的一些问题,其实这些技术无论是客户端还是后端都已经用烂掉了。 ...

  • Android 序列化 Parcelable VS Seria

    Android 的序列化方式 Parcelable Parcel 介绍:Parcel 内部包装了可序列化的数据,可...

  • Bundle

    1.新建一个Bundle类Bundle bundle=new Bundle();2.Bundle类中放入数据(ke...

  • Parcel类详解

    1 Parcel简介 Parcel翻译过来就是打包的意思,其实就是包装了我们需要传输的数据,然后在Binder中传...

  • Binder系列之:Parcel之writeStrongBind

    背景:Parcel.cpp是Android提供的一个容器类,通过binder传输的数据都可以写进Parcel中。但...

  • centos离线安装CDH(4.1): 安装大数据集群

    大数据集群安装配置修改 进入配置界面的方式: 配置parcel的地址 配置parcel地址,能够让我们在后续安装的...

  • Parcel

    两个进程间通信如何传递数据,数据传递和对象的传递是一样的么Parcel就是为了解决这个问题Parcel类似集装箱,...

  • Bundle源码解析

    Bundle源码解析 做一个调用系统分享json的时候遇到一个问题,在用Bundle传递String太大的时候会报...

网友评论

      本文标题:Parcel 解析 Bundle 数据

      本文链接:https://www.haomeiwen.com/subject/iqqxaktx.html