使用Intent传递对象的两种方式

2018-04-07 23:35:16

Intent 的用法相信你已经比较熟悉了,我们可以借助它来启动活动、发送广播、启动服务等。在进行上述操作的时候,我们还可以在Intent 中添加一些附加数据,以达到传值的效果,比如在FirstActivity 中添加如下代码:


[java] view plain copy

  1. Intent intent = new Intent(FirstActivity.this, SecondActivity.class);  

  2. intent.putExtra("string_data""hello");  

  3. intent.putExtra("int_data"100);  

  4. startActivity(intent);  



这里调用了Intent 的putExtra()方法来添加要传递的数据,之后在SecondActivity 中就可以得到这些值了,代码如下所示:

[java] view plain copy

  1. getIntent().getStringExtra("string_data");  

  2. getIntent().getIntExtra("int_data"0);  


但是不知道你有没有发现,putExtra()方法中所支持的数据类型是有限的,虽然常用的一些数据类型它都会支持,但是当你想去传递一些自定义对象的时候就会发现无从下手。不用担心,下面我们就学习一下使用Intent 来传递对象的技巧。


方式一:Serializable 方式

使用Intent 来传递对象通常有两种实现方式,Serializable 和Parcelable,我们先来学习一下第一种的实现方式。
Serializable 是序列化的意思,表示将一个对象转换成可存储或可传输的状态。序列化后的对象可以在网络上进行传输,也可以存储到本地。至于序列化的方法也很简单,只需要让一个类去实现Serializable 这个接口就可以了。
比如说有一个Person 类,其中包含了name 和age 这两个字段,想要将它序列化就可以这样写:


[java] view plain copy

  1. public class Person implements Serializable{  

  2. private String name;  

  3. private int age;  

  4. public String getName() {  

  5.         return name;  

  6.     }  

  7. public void setName(String name) {  

  8.         this.name = name;  

  9.     }  

  10. public int getAge() {  

  11.         return age;  

  12.         }  

  13. public void setAge(int age) {  

  14.         this.age = age;  

  15.     }  

  16. }  



其中get、set 方法都是用于赋值和读取字段数据的,最重要的部分是在第一行。这里让Person 类去实现了Serializable 接口,这样所有的Person 对象就都是可序列化的了。

接下来在FirstActivity 中的写法非常简单:

[java] view plain copy

  1. Person person = new Person();  

  2. person.setName("Tom");  

  3. person.setAge(20);  

  4. Intent intent = new Intent(FirstActivity.this, SecondActivity.class);  

  5. intent.putExtra("person_data", person);  

  6. startActivity(intent);  


可以看到,这里我们创建了一个Person 的实例,然后就直接将它传入到putExtra()方法中了。由于Person 类实现了Serializable 接口,所以才可以这样写。
接下来在SecondActivity 中获取这个对象也很简单,写法如下:


[java] view plain copy

  1. Person person = (Person) getIntent().getSerializableExtra("person_data");  


这里调用了getSerializableExtra()方法来获取通过参数传递过来的序列化对象,接着再将它向下转型成Person 对象,这样我们就成功实现了使用Intent 来传递对象的功能了。



方式二:Parcelable

除了Serializable 之外,使用Parcelable 也可以实现相同的效果,不过不同于将对象进行序列化,Parcelable 方式的实现原理是将一个完整的对象进行分解,而分解后的每一部分都是Intent 所支持的数据类型,这样也就实现传递对象的功能了。
下面我们来看一下Parcelable 的实现方式,修改Person 中的代码,如下所示:


[java] view plain copy

  1. public class Person implements Parcelable {  

  2.     private String name;  

  3.     private int age;  

  4.       

  5.     @Override  

  6.     public int describeContents() {  

  7.         // TODO Auto-generated method stub  

  8.         return 0;  

  9.     }  

  10.   

  11.     @Override  

  12.     public void writeToParcel(Parcel dest, int flags) {  

  13.         // TODO Auto-generated method stub  

  14.         dest.writeString(name);  

  15.         dest.writeInt(age);  

  16.     }  

  17.     public static final Parcelable.Creator<Person> CREATOR=new Parcelable.Creator<Person>() {  

  18.   

  19.         @Override  

  20.         public Person createFromParcel(Parcel source) {  

  21.             // TODO Auto-generated method stub  

  22.             Person person=new Person();  

  23.             person.name=source.readString();  

  24.             person.age=source.readInt();  

  25.             return person;  

  26.         }  

  27.   

  28.         @Override  

  29.         public Person[] newArray(int size) {  

  30.             // TODO Auto-generated method stub  

  31.             return new Person[size];  

  32.         }  

  33.     };  

  34.   

  35. }  


Parcelable 的实现方式要稍微复杂一些。可以看到,首先我们让Person 类去实现了Parcelable 接口,这样就必须重写describeContents()和writeToParcel()这两个方法。其中describeContents()方法直接返回0 就可以了,而writeToParcel()方法中我们需要调用Parcel的writeXxx()方法将Person 类中的字段一一写出。注意字符串型数据就调用writeString()方法,整型数据就调用writeInt()方法,以此类推。
除此之外,我们还必须在Person 类中提供一个名为CREATOR 的常量,这里创建了Parcelable.Creator 接口的一个实现,并将泛型指定为Person。接着需要重写createFromParcel()和newArray()这两个方法,在createFromParcel()方法中我们要去读取刚才写出的name 和age字段,并创建一个Person 对象进行返回,其中name 和age 都是调用Parcel 的readXxx()方法读取到的,注意这里读取的顺序一定要和刚才写出的顺序完全相同。而newArray()方法中的实现就简单多了,只需要new 出一个Person 数组,并使用方法中传入的size 作为数组大小就可以了。
接下来在FirstActivity 中我们仍然可以使用相同的代码来传递Person 对象,只不过在SecondActivity 中获取对象的时候需要稍加改动,如下所示:



[java] view plain copy

  1. Person person = (Person) getIntent().getParcelableExtra("person_data");  


注意这里不再是调用getSerializableExtra()方法,而是调用getParcelableExtra()方法来获取传递过来的对象了,其他的地方都完全相同。这样我们就把使用Intent 来传递对象的两种实现方式都学习完了,对比一下,Serializable的方式较为简单,但由于会把整个对象进行序列化,因此效率方面会比Parcelable 方式低一些,所以在通常情况下还是更加推荐使用Parcelable 的方式来实现Intent 传递对象的功能。


  • 2020-06-02 08:57:12

    clipboard复制成功但是粘贴板是空的

    将文本复制到剪贴板应该不难。配置它不需要几十个步骤,也不需要加载数百KB的js文件 。但最重要的是,它不应该依赖Flash或任何臃肿的框架。这就是clipboard.js存在的原因。

  • 2020-06-04 13:54:21

    vue生成的__ob__: Observer无法解析jsonp

    computed 从vuex获得数据,watch监听数据 然而问题就出现在了监听上,监听不到,给个setTimeOut 1000 就能检测到数据了,不然打印时又数据,用的时候时空的,不知道时什么原因。

  • 2020-06-06 20:22:56

    laravel 接收json串

    在做项目的时候发现 用平时的$request->all() 无法获取到请求值

  • 2020-06-09 08:50:28

    LRU原理以及js实现

    LRU(Least recently used,最近最少使用)算法根据数据的历史访问记录来进行淘汰数据,其核心思想是“如果数据最近被访问过,那么将来被访问的几率也更高”。

  • 2020-06-20 06:31:16

    mac下全局配置adb环境

    不提示“command not found”,而是出现一长串帮助说明,那就证明adb已经配置好了。

  • 2020-06-20 06:31:39

    Android 无线调试手机(WiFi 调试)

    手机需要开启 USB 调试 手机和电脑要在同一个局域网(连接同一个 WiFi) adb connect 连接成功后要拔出 USB 线,不然出现同时连接两个设备的问题 执行命令 ”adb tcpip 6666“ 后可能需要重新开启 USB 调试