Parceler

作者: 总是说下次 | 来源:发表于2016-01-29 01:33 被阅读2113次

标签(空格分隔): Android Serialize 翻译 code_generator
Android的Parcelable代码生成器 原码


[TOC]

介绍

在Android 中,Parcelables是在Contexts之前传递数据非常棒的类序列化方式。和传统的Serialization相比(两者之前的差异),parcelable在进行序列化和反序列化时所需要的时间是传统序列化的十分之一。然而Parcelables有一个严重的瑕疵就是包含大量模板代码。实现一个Parceable,必需实现writeToParcel()createFromParcel() 方法,按同样的顺序读和写转化成Parcel。而且,为了使能获取拆包的类型, Parcelable 必需定义public final static Parcelable.Creator CREATOR<T>变量,明确T的类型。更多关于Pacelable接口参数Parcelable接口的使用
Parceler 是一个代码代码生成器包,用于生成Android Parcelable 模板代码。你不再需要实现Parcelable接口,实现 writeToParcel() 、 createFromParcel() 方法,定义 the public static final CREATOR变量。只需要使用@Parcel注解POJO类,Parcelerw会做上面的工作。Parceler使用the Java JSR-269 Annotation Processor,所以我们不再需要手动运行工具去生成Parcelable代码。仅仅注解Java Bean,编译完成,默认(可配置)情况下Parceler会直接序列化Java Bean域.

@Parcel
public class Example {
    String name;
    int age;

    public Example(){ /*Required empty bean constructor*/ }

    public Example(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public String getName() { return name; }

    public int getAge() { return age; }
}

注意:使用默认的序列策略,不能定义private 变量,因为它会产生反射的性能损失。(定义成private也是能编译和运行的)

能直接引用生成的类Example$Parcelable,或者通过Parcels工具类:

Parcelable wrapped = Parcels.wrap(new Example("Andy", 42));
Example example = Parcels.unwrap(wrapped);
example.getName(); // Andy
example.getAge(); // 42

当然,打包了的Parcelable能直接加入到Android Bundle中在不同Activity之前传递。

Bundle bundle = new Bundle();
bundle.putParcelable("example", Parcels.wrap(example));
Example example = Parcels.unwrap(getIntent().getParcelableExtra("example"));

打包和拆包技术是工厂方式很好实现。此外,Parceler被以下libraries支持。

  • Transfuse - Allows @Parcel annotated beans to be used with the @Extra injection.
  • FragmentArgs - Uses the ParcelerArgsBundler adapter to wrap and unwrap @Parcel annotated beans with fragment parameters.
  • Dart - Autodetects @Parcel annotated beans and automatically unwraps them when using @InjectExtra.

Parcel支持的属性类型

仅有一部分类型能用于注解的@Parcel类的属性,以下列表包括能映射的类型:

  • byte
  • double
  • float
  • int
  • long
  • char
  • boolean
  • String
  • IBinder
  • Bundle
  • SparseArray of any of the mapped types*
  • SparseBooleanArray
  • List, ArrayList and LinkedList of any of the mapped types*
  • Map, HashMap, LinkedHashMap, SortedMap, and TreeMap of any of the mapped types*
  • Set, HashSet, SortedSet, TreeSet, LinkedHashSet of any of the mapped types*
  • Parcelable
  • Serializable
  • Array of any of the mapped types
  • Any other class annotated with @Parcel

如果泛型参数不能被映射Parcel会报错

Parceler 直接支持上面的所有类型,这非常方便在处理集合类:

Parcelable listParcelable = Parcels.wrap(new ArrayList<Example>());
Parcelable mapParcelable = Parcels.wrap(new HashMap<String, Example>());

多态

注意Parceler不能拆包继承关系,所以任何多态域都会被解压成基础类,之所以这样是因为Parceler 选择了性能,并没有校验每份数据.getClass()

@Parcel
public class Example {
    public Parent p;
    //必需添加@ParcelConstructor注解,否则会因为找不到默认构造函数报错
    Example(Parent p) { this.p = p; }
}

@Parcel public class Parent {}
@Parcel public class Child extends Parent {}

Example example = new Example(new Child());
System.out.println(example.p instanceof Child); // true
example = Parcels.unwrap(Parcels.wrap(example));
System.out.println(example.p instanceof Child); // false
/* 说明一下,上面测试代码如果在同一个context里面则两个都是true,看Parcels.unwrap就知道,他是直接获取的之前的对象,中间并没有打包和拆包的过程。*/

参考自定义序列化部分,实现多态域。

序列化策略

Parceler 除了上面以基本域的序列化方法外,还提供几种选择,如何去序列化和反序列化一个对象。

Getter/setter serialization

Parceler能配置序列化使用的gettersetternon-empty constructor.另外,field可通过@ParcelProperty注解与方法和构造函数的参数关联,这将支持很多bean,包括不可变的bean
配置使用getter/settter方法序列化,只需简单给@Parcel注解设置其value值为Serialization.BEAN

@Parcel(Serialization.BEAN)
public class Example {
    private String name;
    private int age;

    public String getName() { return name; }
    public void setName(String name) { this.name = name; }

    public int getAge() { return age; }
    public void setAge(int age) { this.age = age; }
}

使用带参构造函数序列化,必需在带参函数上添加 @ParcelConstructor注解:

@Parcel(Serialization.BEAN)
public class Example {
    private final String name;
    private final int age;

    @ParcelConstructor
    public Example(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public String getName() { return name; }

    public int getAge() { return age; }
}

默认会使用无参构造函数,除非带参构造函数有注解。如果没有无参构造函数且没@ParentConstructor注解的带参构造函数,编译无法通过。

getters/setters and fields混合使用

你能混合使用序列化策略,使用@ParcelProperty。下面的例子中firstName和lastName使用构造函数打包,然而拆包fistName通过引用直接读取,而lastName则通过getLastName()方法读取。first、last作为@ParcelProperty()标识,必需成对出现 。

@Parcel
public class Example {
    //关联使用该注解的属性
    @ParcelProperty("first")
    String firstName;
    String lastName;

    @ParcelConstructor
    public Example(@ParcelProperty("first") String firstName, @ParcelProperty("last") String lastName){
        this.firstName = firstName;
        this.lastName = lastName;
    }

    public String getFirstName() { return firstName; }

    @ParcelProperty("last")
    public String getLastName() { return lastName; }
}

不需要序列化的属性能在其getter/settter方法上注解@Transient,另外带transient关键字的域也不会被序列化。

Parceler 支持不同种类的POJO,允许@Parcel注解的类用于其它处理POJO的包,包括以下

静态工厂支持

直接使用构造函数是可选的,Parceler支持使用注解静态工厂创建给定类的实例。这种实现支持Google的AutoValue注解处理器生成不可变beans. 实现了AutoValue的抽象类,通过 @ParcelFactory 注解工厂方法,映射成有@Parcel的类。
使用@AutoValue可能会报空指针异常,问题可参考https://github.com/google/auto/issues/240

@AutoValue
@Parcel
public abstract class AutoValueParcel {

    @ParcelProperty("value") public abstract String value();

    @ParcelFactory
    public static AutoValueParcel create(String value) {
        return new AutoValue_AutoValueParcel(value);
    }
}

AutoValue 能生成同的类,因此告诉Parcles工具类应该实例化哪个类:

Parcelable wrappedAutoValue = Parcels.wrap(AutoValueParcel.class, AutoValueParcel.create("example"));

反序列化

AutoValueParcel autoValueParcel = Parcels.unwrap(wrappedAutoValue);

自定义序列化

@Parcel可通过@ParcelPropertyConverter(ParcelConverter)去指定任意一个field序列化过程.
下面例子演示了使用ParcelConverter反序化有继承关系的类

@Parcel
public class Item {
    @ParcelPropertyConverter(ItemListParcelConverter.class)
    public List<Item> itemList;
}
@Parcel public class SubItem1 extends Item {}
@Parcel public class SubItem2 extends Item {}

public class ItemListParcelConverter implements ParcelConverter<List<Item>> {
    @Override
    public void toParcel(List<Item> input, Parcel parcel) {
        if (input == null) {
            parcel.writeInt(-1);
        }
        else {
            parcel.writeInt(input.size());
            for (Item item : input) {
                parcel.writeParcelable(Parcels.wrap(item), 0);
            }
        }
    }

    @Override
    public List<Item> fromParcel(Parcel parcel) {
        int size = parcel.readInt();
        if (size < 0) return null;
        List<Item> items = new ArrayList<Item>();
        for (int i = 0; i < size; ++i) {
            items.add((Item) Parcels.unwrap(parcel.readParcelable(Item.class.getClassLoader())));
        }
        return items;
    }
}

Parceler在api的org.parcler.converter包下实现一系列对集体的转换,这些类处理了一些判空和迭代器,上面的ParcelConverter可以简单的实现:

public class ItemListParcelConverter extends ArrayListParcelConverter<Item> {
    @Override
    public void itemToParcel(Item item, Parcel parcel) {
        parcel.writeParcelable(Parcels.wrap(item), 0);
    }

    @Override
    public Item itemFromParcel(Parcel parcel) {
        return Parcels.unwrap(parcel.readParcelable(Item.class.getClassLoader()));
    }
}

没有source 的 classes

对于那些没有Java source 的classses ,也能 通过使用@ParcelClass注解进行序列化。这个注解能方便有用于可编译资源的任何地方。例如,在Android Appliction 中可对LibarayParcel.class 序列化:

@ParcelClass(LibraryParcel.class)
public class AndroidApplication extends Application{
    //...
}

序列化多个类时使用@ParcelClasses注解

另外,@ParcelClass关联的类可以annotation属性配置其将要使用的@Parcel注解的参数,这样我们就能配置关联类的序列化策略,以及哪些类需要分析。
一个有用的策略,规定一个类型使用自定的converters.

@ParcelClass(
    value = LibraryParcel.class,
    annotation = @Parcel(converter = LibraryParcelConverter.class))
class SomeClass{}

这将在class的基础上进行控制,所以不能再动态更改。

高级配置

跳过分析

libraries中可能需要bean去扩展一个base class,并要求基类中的数据不需要打包。Parceler可通过analysis配置该类中的哪基类类数据打包:

@Parcel(analyze = {One.class, Three.class})
class One extends Two {}
class Two extends Three {}
class Three extends BaseClass {}

这个例子,只有OneThree会被序列化,避开了TwoBaseClass的参数

定义打包过程

Parcels工具类分类查找所有需要打包的类,由于性能优化的原因它直接忽略了class的继承关系,无法对@Parcel的bean的子类进行打包.有两种方法可解决这个问题:
第一种,通过implementations参数去关系他的子类:

class ExampleProxy extends Example {}
@Parcel(implementations = {ExampleProxy.class})
class Example {}

ExampleProxy proxy = new ExampleProxy();
Parcels.wrap(proxy);  // ExampleProxy will be serialized as a Example

第二种,在使用Parcels.wrap()方法时,指定要打包的类(已注解@Parcel,并且是继承关系)

ExampleProxy proxy = new ExampleProxy();
Parcels.wrap(Example.class, proxy);

避免indexing错误

Libraries中使用Parceler将会面临挑战,因为Parceler生成唯一个映射类Parceler$$Parcels去关联一个bean和bean所对应的Parcelable,编译时这个映射类可能冲突,并会抛出下面的错误:

Error Code:
    2
Output:
    UNEXPECTED TOP-LEVEL EXCEPTION:
    com.android.dex.DexException: Multiple dex files define Lorg/parceler/Parceler$$Parcels$1;
        at com.android.dx.merge.DexMerger.readSortableTypes(DexMerger.java:594)
        at com.android.dx.merge.DexMerger.getSortedTypes(DexMerger.java:552)
        at com.android.dx.merge.DexMerger.mergeClassDefs(DexMerger.java:533)
        ....

为了避免去这个公共的映射类,设置parcelsIndex = false到每一个使用@Parcel注解的类,Parceler将不会生成Parceler$$Parcels映射类,若没有没映射做为索引,Parceler会退回去根据类名查找生成的class
或者,在顶层项目使用@ParcelClass配置,而不需要在每@Parcel注解上进行配置。

配置混淆

对于使用了代码混淆,添加以下几行到你的混淆配置文件里。这将保留Parcels 相关工具类以及Parcelable CARETOR [静态工厂]实例

# Parcel library
-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}

-keep class org.parceler.Parceler$$Parcels

引入Parceler

Maven

<dependency>
    <groupId>org.parceler</groupId>
    <artifactId>parceler</artifactId>
    <version>1.0.4</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.parceler</groupId>
    <artifactId>parceler-api</artifactId>
    <version>1.0.4</version>
</dependency>

Gradle

compile "org.parceler:parceler-api:1.0.4"
apt "org.parceler:parceler:1.0.4"

详细使用请参考android-apt project

相关文章

网友评论

    本文标题:Parceler

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