Intent 的用法相信你已经比较熟悉了,我们可以借助它来启动活动、发送广播、启动服务等。在进行上述操作的时侯,我们还可以在 Intent 中添加一些附加数据,以达到传值的效果,比如在 FirstActivity 中添加如下代码。
val intent = Intent(this,SecondActivity::class.java)
intent.putExtra("string_data","hello")
intent.putExtra("int_data",100)
startActivity(intent)
这里调用了 Intent 的 putExtra() 方法来添加要传递的数据,之后在 SecondActivity 中就可以得到这些值了,代码如下所示:
intent.getStringExtra("string_data")
intent.getIntExtra("int_data",0)
但是不知道你有没有发现,putExtra() 方法中所支持的数据类型是有限的。虽然常用的一些数据类型是支持的,但是当你想去传递一些自定义对象的时候,就会发现无从下手。不用担心,下面我们就学习一下使用Intent 来传递对象的技巧。
14.2.1 Serializable 方式
使用 Intent 来传递对象通常有两种实现方式:Serializable 和 Parcelablc,本小节中我们先来学习一下第一种实现方式。
Serializable 是序列化的意思,表示将一个对象转换成可存储或可传输的状态。序列化后的对象可以在网络上进行传输,也可以存储到本地。至于序列化的方法也很简单,只需要让一个类去实现 Serializable 这个接口就可以了。
比如说有一个 Person 类,其中包含了 name 和 age 这两个字段,想要将它序列化就可以这样写:
class Person: Serializable{
var name = ""
var age = 0
}
这里我们让Person 类实现了Serializable 接口,这样所有的Person 对象都是可序列化的了。
然后在FirstActivity 中只需要这样写:
val person = Person()
person.name = "Tom"
person.age = 20
val intent = Intent(this,SecondActivity::class.java)
intent.putExtra("person_data",person)
startActivity(intent)
可以看到,这里我们创建了一个 Person 的实例,然后就直接将它传人到 putExtra() 方法中了。由于 Person 类实现了 Serializable 接口,所以才可以这样写。
接下来在 SecondActivity 中获取这个对象也很简单,写法如下:
val person = intent.getSerializableExtra("person_data") as Person
这里调用了 getSerializableExtra() 方法来状取通过参数传递过来的序列化对象,接着再将它向下转型成 Person 对象,这样我们就成功实现了使用 Intent 来传递对象的功能了。
需要注意的是,这种传递对象的工作原理是先将一个对象序列化成可存储或可传输的状态,传递给另一个Activity 后再将其反序列化成一个新的对象。虽然这两个对象中存储的数据完全一致,但是它们实际上是不同的对象,这一点希望你能了解清楚。
14.2.2 Parcelable 方式
除了Serializable 之外,使用Parcelable 也可是实现相同的想过。不过不同于将对象进行序列化,Parcelable 方式的实现原理是将一个完整的对象进行分解,而分解后的每一部分都是Intent 所支持的数据类型,这样就能实现传递对象的功能了。
下面我们来看一下 Parcelable 的实现方式,修改 Person 中的代码,如下所示:
class Person :Parcelable {
var name = ""
var age = 0
override fun writeToParcel(parcel: Parcel, flags: Int) {
parcel.writeString(name) // 写出name
parcel.writeInt(age) // 写出age
}
override fun describeContents(): Int {
return 0
}
companion object CREATOR : Parcelable.Creator {
override fun createFromParcel(parcel: Parcel): Person {
val person = Person()
person.name = parcel.readString() ?: "" // 读取name
person.age = parcel.readInt() ?: 0 // 读取age
return person
}
override fun newArray(size: Int): Array {
return arrayOfNulls(size)
}
}
}
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 中获取对象的时候需要稍加改动,如下所示:
val person = intent.getParcelableExtra("person_data") as Person
注意,这里不再是调用 getSerializableExtra() 方法,而是调用 getParcelableExtra() 方法来获取传递过来的对象了,其他的地方都完全相同。
不过,这种实现方式写起来确实比较复杂,为此Kotlin 给我们提供了另外一个更加简明的用法,但前提出要传递的所有数据都必须封装在对象的主构造函数中才行。
修改Person 类中的代码,如下所示:
@Parcelize
class Person(var name: String,var age: Int):Parcelable
没错,就是这么简单。将name 和 age 这两个字段移动到主构造函数中,然后给Person 类添加一个@Parcelize 注解即可,是不是比之前的用法简单了好多倍?
这样我们就把使用 Intent 来传递对象的两种实现方式都学习完了,对比一下,Serializable 的方式较为简单,但由于会把整个对象进行序列化,因此效率会比 Parcelable 方式低一些,所以在通常情况下还是更加推荐使用 Parcelable 的方式来实现 Intent 传递对象的功能。