利用Kotlin如何實(shí)現(xiàn)Android開發(fā)中的Parcelable詳解
坑
先來看看 Android Studio 給的自動(dòng)實(shí)現(xiàn)。
新建一個(gè)數(shù)據(jù)類,讓它實(shí)現(xiàn) Parcelable
data class Worker( var id: Int, var name: String, var tasks: MutableList<Int> ) : Parcelable
使用 Android Studio 自帶的 Add Parcelable Implementation ,然后你就得到了。。。
data class Worker( var id: Int, var name: String, var tasks: MutableList<Int> ) : Parcelable { constructor(parcel: Parcel) : this( parcel.readInt(), parcel.readString(), TODO("tasks")) { } override fun writeToParcel(parcel: Parcel, flags: Int) { parcel.writeInt(id) parcel.writeString(name) } override fun describeContents(): Int { return 0 } companion object CREATOR : Parcelable.Creator<Worker> { override fun createFromParcel(parcel: Parcel): Worker { return Worker(parcel) } override fun newArray(size: Int): Array<Worker?> { return arrayOfNulls(size) } } }
有什么問題呢?
至少現(xiàn)在可以編譯過了 。。。
很明顯的,自動(dòng)生成的 Parcelable 實(shí)現(xiàn)沒有包含對(duì) MutableList 的處理,因?yàn)?Parcel 原生只支持 ArrayList ,所以這是需要你自己實(shí)現(xiàn)的部分。先來解決這個(gè)問題。
雖然名字是 MutableList ,但是實(shí)際上這只是 Kotlin 的一個(gè)輔助類型,可以用 Tools -> Kotlin -> Show Kotlin Bytecode 查看它編譯成 JVM 字節(jié)碼之后的樣子。
// access flags 0x2 // signature Ljava/util/List<Ljava/lang/Integer;>; // declaration: java.util.List<java.lang.Integer> private Ljava/util/List; tasks @Lorg/jetbrains/annotations/NotNull;() // invisible
點(diǎn)擊 [Decompile] 按鈕還可以直接反編譯到 Java 。
編譯之后 MutableList 變成了 Java 的原生類型 java.util.List 。因此我們只需要在對(duì)應(yīng)的地方調(diào)用 Parcel 中對(duì) List 和 ArrayList 的處理方法就可以了。
constructor(parcel: Parcel) : this( parcel.readInt(), parcel.readString(), parcel.readArrayList(Int::class.java.classLoader) as MutableList<Int>) { } override fun writeToParcel(parcel: Parcel, flags: Int) { parcel.writeInt(id) parcel.writeString(name) parcel.writeList(tasks) }
writeList 是可以兼容 Kotlin 的 List 與 MutableList 類型的,但是 ArrayList 還需要強(qiáng)轉(zhuǎn)一下才行,雖然能跑但是會(huì)很難看,能不能變好看一點(diǎn)呢?
加一個(gè)擴(kuò)展方法就好了
inline fun <reified T> Parcel.readMutableList(): MutableList<T> { @Suppress("UNCHECKED_CAST") return readArrayList(T::class.java.classLoader) as MutableList<T> }
然后就可以這樣寫
constructor(parcel: Parcel) : this( parcel.readInt(), parcel.readString(), parcel.readMutableList()) { } override fun writeToParcel(parcel: Parcel, flags: Int) { parcel.writeInt(id) parcel.writeString(name) parcel.writeList(tasks) }
CREATOR 與 companion object 之爭(zhēng)
Parcelable 有個(gè)特殊的要求,在 Android 官方文檔 里是這樣寫的
Parcelable interface must also have a non-null static field called CREATOR of a type that implements the Parcelable.Creator interface.
這是因?yàn)?Java 的泛型有運(yùn)行時(shí)消除機(jī)制的限制, Parcel 需要一個(gè)輔助對(duì)象來協(xié)助構(gòu)造你的對(duì)象以及你的對(duì)象的數(shù)組,這就是 CREATOR 。 Parcelable 要求每個(gè)實(shí)現(xiàn)類都有這個(gè) CREATOR 對(duì)象,并且它必須是非空的、公有的、靜態(tài)字段。在 Java 程序中,對(duì)于每個(gè)類 CREATOR 有非常穩(wěn)定的實(shí)現(xiàn)。假如上面的例子是用 Java 寫的,由于我們已經(jīng)有了一個(gè)以 Parcel 為參數(shù)的構(gòu)造方法,我們只需要這樣實(shí)現(xiàn) CREATOR 。
public static final Creator<Worker> CREATOR = new Creator<Worker>() { @Override public Worker createFromParcel(Parcel in) { return new Worker(in); } @Override public Worker[] newArray(int size) { return new Worker[size]; } };
那么在 Kotlin 中是什么樣的呢,我們可以先看看 Android Studio 生成的實(shí)現(xiàn):
companion object CREATOR : Parcelable.Creator<Worker> { override fun createFromParcel(parcel: Parcel): Worker { return Worker(parcel) } override fun newArray(size: Int): Array<Worker?> { return arrayOfNulls(size) } }
在 Kotlin 中,使用命名的 companion object 確實(shí)可以生成一個(gè)對(duì)應(yīng)名字的靜態(tài)字段,并且它是公有的,會(huì)隨著類的加載而被創(chuàng)建。但是一個(gè)類里只能有一個(gè)伴生對(duì)象,這個(gè)實(shí)現(xiàn)把伴生對(duì)象給占據(jù)了。雖然并沒有什么影響的樣子,但是看著總是不舒服。
通過 Kotlin 提供的 @JvmField 注解,我們可以讓 Kotlin 編譯器把它作為一個(gè)字段進(jìn)行處理,那我們可以在 companion object 里定義一個(gè) CREATOR ,然后給它加上 @JvmField 注解。
companion object { @JvmField val CREATOR = object : Parcelable.Creator<Worker> { override fun createFromParcel(parcel: Parcel): Worker { return Worker(parcel) } override fun newArray(size: Int): Array<Worker?> { return arrayOfNulls(size) } } }
這樣做有什么好處呢? CREATOR 不再占據(jù)整個(gè) companion object ,而是只是作為 companion object 中的一個(gè)字段,代碼干凈了很多。
此外, Kotlin 還對(duì) inline 方法提供了 reified 泛型機(jī)制,這種泛型會(huì)被編譯器直接具體化而不會(huì)像 Java 泛型一樣會(huì)被運(yùn)行時(shí)擦除。如果不需要太考慮效率,我們可以定義一個(gè)這樣的方法。
inline fun <reified T : Parcelable> parcelableCreatorOf(): Parcelable.Creator<T> = object : Parcelable.Creator<T> { override fun newArray(size: Int): Array<T?> = arrayOfNulls(size) override fun createFromParcel(source: Parcel?): T = T::class.java.getDeclaredConstructor(Parcel::class.java).newInstance(source) }
在每一個(gè) Parcelable 實(shí)現(xiàn)類中就只需要一行代碼了。
companion object { @JvmField val CREATOR = parcelableCreatorOf<Worker>() }
End
最后,再來看看我們的 Parcelable 實(shí)現(xiàn)類。
data class Worker( var id: Int, var name: String, var tasks: MutableList<Int> ) : Parcelable { constructor(parcel: Parcel) : this( parcel.readInt(), parcel.readString(), parcel.readMutableList()) override fun writeToParcel(parcel: Parcel, flags: Int) { parcel.writeInt(id) parcel.writeString(name) parcel.writeList(tasks) } override fun describeContents(): Int = 0 companion object { @JvmField val CREATOR = parcelableCreatorOf<Worker>() } }
本文中的關(guān)鍵代碼,我已經(jīng)封裝成了一個(gè)工具類,添加依賴即可使用 -> KotlinUtils
Kotlin使用parcelable出現(xiàn):BadParcelableException: Parcelable protocol requires a Parcelable.Creator...
在Kotlin編寫代碼過程中,需要用到parcelable來進(jìn)行傳值,按照以前的寫法,進(jìn)行序列化:
class PayTypeInfo : Parcelable{ var payMethodId: String? = null//支付方式ID var payMethodName: String? = null//支付方式名稱 override fun writeToParcel(dest: Parcel, flags: Int) { dest.writeString(payMethodId) dest.writeString(payMethodName) } override fun describeContents(): Int { return 0 } companion object { val CREATOR: Parcelable.Creator<PayTypeInfo> = object : Parcelable.Creator<PayTypeInfo> { override fun createFromParcel(source: Parcel): PayTypeInfo { val payTypeInfo = PayTypeInfo() payTypeInfo.payMethodId = source.readString() payTypeInfo.payMethodName = source.readString() return payTypeInfo } override fun newArray(size: Int): Array<PayTypeInfo> { return newArray(size) } } } }
這樣序列化的實(shí)體類就寫完了,然后進(jìn)行傳值
val bundle = Bundle() val typeList = ArrayList<PayTypeInfo>() bundle.putParcelableArrayList("payType", typeList)
接受數(shù)據(jù)時(shí):
val bundle = intent.extras val payTypeList = bundle.getParcelableArrayList<PayTypeInfo>("payType")
運(yùn)行程序,出現(xiàn)錯(cuò)誤,錯(cuò)誤代碼為:BadParcelableException: Parcelable protocol requires a Parcelable.Creator...
經(jīng)過查找資料,找到了解決辦法,只需要在代碼CREATOR前面添加@JvmField即可:
@JvmField val CREATOR: Parcelable.Creator<PayTypeInfo> = object : Parcelable.Creator<PayTypeInfo> { override fun createFromParcel(source: Parcel): PayTypeInfo { val payTypeInfo = PayTypeInfo() payTypeInfo.payMethodId = source.readString() payTypeInfo.payMethodName = source.readString() return payTypeInfo } override fun newArray(size: Int): Array<PayTypeInfo> { return newArray(size) } }
在運(yùn)行程序,傳值成功
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Android創(chuàng)建淡入淡出動(dòng)畫的詳解
大家好,本篇文章主要講的是Android創(chuàng)建淡入淡出動(dòng)畫的詳解,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12Fiddler實(shí)現(xiàn)手機(jī)抓包之小白入門必看
這篇文章主要介紹了Fiddler實(shí)現(xiàn)手機(jī)抓包之小白入門必看篇,需要的朋友可以參考下2018-03-03Android adb.exe程序啟動(dòng)不起來 具體解決方法
這篇文章主要介紹了Android adb.exe程序啟動(dòng)不起來 具體解決方法,有需要的朋友可以參考一下2013-12-12基于Flutter實(shí)現(xiàn)轉(zhuǎn)場(chǎng)動(dòng)效的示例代碼
動(dòng)畫經(jīng)常會(huì)用于場(chǎng)景切換,比如滑動(dòng),縮放,尺寸變化。Flutter?提供了Transition系列的動(dòng)畫組件,可以讓場(chǎng)景轉(zhuǎn)換動(dòng)畫變得更加簡(jiǎn)單。本文整理了常用的Transition組件的應(yīng)用,需要的可以參考一下2022-05-05解決Android Studio 出現(xiàn)“Cannot resolve symbo
今天在調(diào)試的時(shí)候,Android Studio報(bào)了一個(gè)莫名其妙的錯(cuò)誤Cannot resolve symbol'R'讓人不知所措,因?yàn)檫@東西根本不歸我管啊,怎么會(huì)出現(xiàn) Cannot resolve symbol 這種錯(cuò)誤呢?下面給大家分享Android Studio 出現(xiàn)“Cannot resolve symbol”解決方案,需要的朋友可以參考下2023-03-03詳解Android中fragment和viewpager的那點(diǎn)事兒
本文主要對(duì)Android中fragment和viewpager進(jìn)行詳細(xì)介紹,具有一定的參考價(jià)值,需要的朋友一起來看下吧2016-12-12Android開發(fā)之項(xiàng)目模塊化實(shí)踐教程
這篇文章主要給大家介紹了關(guān)于Android開發(fā)之項(xiàng)目模塊化的相關(guān)資料,文中通過示例代碼給各位Android開發(fā)者們介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)下吧。2017-09-09Android拖動(dòng)條的實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了Android拖動(dòng)條的實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09Android開發(fā)實(shí)現(xiàn)布局中為控件添加選擇器的方法
這篇文章主要介紹了Android開發(fā)實(shí)現(xiàn)布局中為控件添加選擇器的方法,涉及Android開發(fā)中布局設(shè)置的相關(guān)操作技巧,需要的朋友可以參考下2017-10-10