真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

利用Kotlin如何實(shí)現(xiàn)Android開(kāi)發(fā)中的Parcelable詳解

站在用戶(hù)的角度思考問(wèn)題,與客戶(hù)深入溝通,找到陵水黎族網(wǎng)站設(shè)計(jì)與陵水黎族網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶(hù)體驗(yàn)好的作品,建站類(lèi)型包括:網(wǎng)站制作、成都網(wǎng)站制作、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名注冊(cè)、雅安服務(wù)器托管、企業(yè)郵箱。業(yè)務(wù)覆蓋陵水黎族地區(qū)。

先來(lái)看看 Android Studio 給的自動(dòng)實(shí)現(xiàn)。

新建一個(gè)數(shù)據(jù)類(lèi),讓它實(shí)現(xiàn) Parcelable

data class Worker(
  var id: Int,
  var name: String,
  var tasks: MutableList
) : Parcelable

使用 Android Studio 自帶的 Add Parcelable Implementation ,然后你就得到了。。。

 data class Worker(
  var id: Int,
  var name: String,
  var tasks: MutableList
) : 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 {
  override fun createFromParcel(parcel: Parcel): Worker {
   return Worker(parcel)
  }
  override fun newArray(size: Int): Array {
   return arrayOfNulls(size)
  }
 }
}

有什么問(wèn)題呢?

至少現(xiàn)在可以編譯過(guò)了 。。。

很明顯的,自動(dòng)生成的 Parcelable 實(shí)現(xiàn)沒(méi)有包含對(duì) MutableList 的處理,因?yàn)?Parcel 原生只支持 ArrayList ,所以這是需要你自己實(shí)現(xiàn)的部分。先來(lái)解決這個(gè)問(wèn)題。

雖然名字是 MutableList ,但是實(shí)際上這只是 Kotlin 的一個(gè)輔助類(lèi)型,可以用 Tools -> Kotlin -> Show Kotlin Bytecode 查看它編譯成 JVM 字節(jié)碼之后的樣子。

// access flags 0x2
// signature Ljava/util/List;
// declaration: java.util.List
private Ljava/util/List; tasks
@Lorg/jetbrains/annotations/NotNull;() // invisible

點(diǎn)擊 [Decompile] 按鈕還可以直接反編譯到 Java 。

編譯之后 MutableList 變成了 Java 的原生類(lèi)型 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) {
}
override fun writeToParcel(parcel: Parcel, flags: Int) {
 parcel.writeInt(id)
 parcel.writeString(name)
 parcel.writeList(tasks)
}

writeList 是可以兼容 Kotlin 的 List 與 MutableList 類(lèi)型的,但是 ArrayList 還需要強(qiáng)轉(zhuǎn)一下才行,雖然能跑但是會(huì)很難看,能不能變好看一點(diǎn)呢?

加一個(gè)擴(kuò)展方法就好了

inline fun  Parcel.readMutableList(): MutableList {
 @Suppress("UNCHECKED_CAST")
 return readArrayList(T::class.java.classLoader) as MutableList
}

然后就可以這樣寫(xiě)

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 官方文檔 里是這樣寫(xiě)的

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ì)象來(lái)協(xié)助構(gòu)造你的對(duì)象以及你的對(duì)象的數(shù)組,這就是 CREATOR 。 Parcelable 要求每個(gè)實(shí)現(xiàn)類(lèi)都有這個(gè) CREATOR 對(duì)象,并且它必須是非空的、公有的、靜態(tài)字段。在 Java 程序中,對(duì)于每個(gè)類(lèi) CREATOR 有非常穩(wěn)定的實(shí)現(xiàn)。假如上面的例子是用 Java 寫(xiě)的,由于我們已經(jīng)有了一個(gè)以 Parcel 為參數(shù)的構(gòu)造方法,我們只需要這樣實(shí)現(xiàn) CREATOR 。

public static final Creator CREATOR = new Creator() {
 @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 {
 override fun createFromParcel(parcel: Parcel): Worker {
  return Worker(parcel)
 }
 override fun newArray(size: Int): Array {
  return arrayOfNulls(size)
 }
}

在 Kotlin 中,使用命名的 companion object 確實(shí)可以生成一個(gè)對(duì)應(yīng)名字的靜態(tài)字段,并且它是公有的,會(huì)隨著類(lèi)的加載而被創(chuàng)建。但是一個(gè)類(lèi)里只能有一個(gè)伴生對(duì)象,這個(gè)實(shí)現(xiàn)把伴生對(duì)象給占據(jù)了。雖然并沒(méi)有什么影響的樣子,但是看著總是不舒服。

通過(guò) Kotlin 提供的 @JvmField 注解,我們可以讓 Kotlin 編譯器把它作為一個(gè)字段進(jìn)行處理,那我們可以在 companion object 里定義一個(gè) CREATOR ,然后給它加上 @JvmField 注解。

companion object {
 @JvmField val CREATOR = object : Parcelable.Creator {
  override fun createFromParcel(parcel: Parcel): Worker {
   return Worker(parcel)
  }
  override fun newArray(size: Int): Array {
   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  parcelableCreatorOf(): Parcelable.Creator = object : Parcelable.Creator {
 override fun newArray(size: Int): Array = arrayOfNulls(size)
 override fun createFromParcel(source: Parcel?): T =
   T::class.java.getDeclaredConstructor(Parcel::class.java).newInstance(source)
}

在每一個(gè) Parcelable 實(shí)現(xiàn)類(lèi)中就只需要一行代碼了。

companion object {
 @JvmField val CREATOR = parcelableCreatorOf()
}

End

最后,再來(lái)看看我們的 Parcelable 實(shí)現(xiàn)類(lèi)。

data class Worker(
  var id: Int,
  var name: String,
  var tasks: MutableList
) : 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()
 }
}

本文中的關(guān)鍵代碼,我已經(jīng)封裝成了一個(gè)工具類(lèi),添加依賴(lài)即可使用 -> KotlinUtils

Kotlin使用parcelable出現(xiàn):BadParcelableException: Parcelable protocol requires a Parcelable.Creator...

在Kotlin編寫(xiě)代碼過(guò)程中,需要用到parcelable來(lái)進(jìn)行傳值,按照以前的寫(xiě)法,進(jìn)行序列化:

class PayTypeInfo : Parcelable{
var payMethodId: String? = null//支付方式ID
var payMethodName: String? = null//支付方式名稱(chēng)
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 = object : Parcelable.Creator {
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 {
return newArray(size)
}
}
}
}

這樣序列化的實(shí)體類(lèi)就寫(xiě)完了,然后進(jìn)行傳值

val bundle = Bundle()
val typeList = ArrayList()
bundle.putParcelableArrayList("payType", typeList)

接受數(shù)據(jù)時(shí):

val bundle = intent.extras
val payTypeList = bundle.getParcelableArrayList("payType")

運(yùn)行程序,出現(xiàn)錯(cuò)誤,錯(cuò)誤代碼為:BadParcelableException: Parcelable protocol requires a Parcelable.Creator...

經(jīng)過(guò)查找資料,找到了解決辦法,只需要在代碼CREATOR前面添加@JvmField即可:

@JvmField val CREATOR: Parcelable.Creator = object : Parcelable.Creator {
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 {
return newArray(size)
}
}

在運(yùn)行程序,傳值成功

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)創(chuàng)新互聯(lián)的支持。


文章名稱(chēng):利用Kotlin如何實(shí)現(xiàn)Android開(kāi)發(fā)中的Parcelable詳解
本文鏈接:http://weahome.cn/article/jhgdjs.html

其他資訊

在線咨詢(xún)

微信咨詢(xún)

電話咨詢(xún)

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部