亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Kotlin 基礎(chǔ)教程之泛型

 更新時(shí)間:2017年06月30日 11:27:18   投稿:lqh  
這篇文章主要介紹了Kotlin 基礎(chǔ)教程之泛型的相關(guān)資料,需要的朋友可以參考下

Kotlin 支持泛型, 語(yǔ)法和 Java 類(lèi)似。

例如,泛型類(lèi):

class Hello<T>(val value: T)

val box = Box<Int>(1)
val box1 = Box(2)

泛型函數(shù):

fun <T> foo(item: T): List<T> {
  // do something
}

val list = foo<Int>(1)

fun <T> T.toString2(): String {
  // 擴(kuò)展函數(shù)
}

fun <K, V>put(key: K, value: V) {
  // 多個(gè)泛型參數(shù)
}

類(lèi)型變異

Java 的泛型中,最難理解的就是通配符。Java 中使用通配符是由于泛型類(lèi)型是不可變的,比如 List<String>不是List<Object>的子類(lèi), 因而 List<Object> objs = strs 這樣的代碼有編譯錯(cuò)誤。

為了解決此問(wèn)題,Java 提供了通配符類(lèi)型參數(shù)(wildcard type argument)。如果你只能從一個(gè)集合取得元素, 那么就可以使用一個(gè) String 組成的集合, 并從中讀取 Object 實(shí)例,這個(gè)時(shí)候用? extends T. 反過(guò)來(lái), 如果你只能向集合 放入 元素, 那么就可以使用一個(gè) Object 組成的集合, 并向其中放入 String, 這個(gè)時(shí)候用? super T。

Kotlin 不存在這樣的通配符,提供了兩種方法:聲明處類(lèi)型變異(declaration-sitevariance), 以及類(lèi)型投射(type projection)。

假設(shè)我們有一個(gè)泛型接口 Source<T> , 其中不存在任何接受 T 作為參數(shù)的方法, 僅有返回值為 T 的方法:

// Java
interface Source<T> {
  T nextT();
}

void demo(Source<String> strs) {
  Source<Object> objects = strs; // !!! 在 Java 中禁止這樣的操作
  // ...
}

為了解決這個(gè)問(wèn)題, 我們不得不將對(duì)象類(lèi)型聲明為 Source<? extends Object> , 其實(shí)是毫無(wú)意義的, 編譯器并不理解這一點(diǎn)。

在 Kotlin 中, 我們有辦法將這種情況告訴編譯器. 這種技術(shù)稱為聲明處的類(lèi)型變異(declaration-sitevariance): 我們可以對(duì) Source 的 類(lèi)型參數(shù) T 添加注解, 來(lái)確保 Source<T> 的成員函數(shù)只會(huì)返回T 類(lèi)型, 而絕不會(huì)消費(fèi) T 類(lèi)型. 為了實(shí)現(xiàn)這個(gè)目的, 我們可以對(duì) T 添加 out 修飾符:

abstract class Source<out T> {
  abstract fun nextT(): T
}
fun demo(strs: Source<String>) {
  val objects: Source<Any> = strs // 這是 OK 的, 因?yàn)?T 是一個(gè) out 類(lèi)型參數(shù)
  // ...
}

一般規(guī)則是: 當(dāng) C 類(lèi)的類(lèi)型參數(shù) T 聲明為 out 時(shí), 那么在 C 的成員函數(shù)中, T 類(lèi)型只允許出現(xiàn)在輸出位置, 這樣的限制帶來(lái)的回報(bào)就是, C<Base> 可以安全地用作 C<Derived> 的父類(lèi)型。

除了 out 之外, Kotlin 還提供了另一種類(lèi)型變異注解: in. 這個(gè)注解導(dǎo)致類(lèi)型參數(shù)反向類(lèi)型變異(contravariant): 這個(gè)類(lèi)型將只能被消費(fèi), 而不能被生產(chǎn). 反向類(lèi)型變異的一個(gè)很好的例子是 Comparable :

abstract class Comparable<in T> {
  abstract fun compareTo(other: T): Int
}

fun demo(x: Comparable<Number>) {
  x.compareTo(1.0) // 1.0 類(lèi)型為 Double, 是 Number 的子類(lèi)型
  // 因此, 我們可以將 x 賦值給 Comparable<Double> 類(lèi)型的變量
  val y: Comparable<Double> = x // OK!
}

類(lèi)型投射(Type projection)

class Array<T>(val size: Int) {
  fun get(index: Int): T { /* ... */ }
  fun set(index: Int, value: T) { /* ... */ }
}

這個(gè)類(lèi)對(duì)于類(lèi)型參數(shù) T 既不能協(xié)變, 也不能反向協(xié)變. 這就帶來(lái)很大的不便。

fun copy(from: Array<Any>, to: Array<Any>) {
  assert(from.size == to.size)
  for (i in from.indices)
    to[i] = from[i]
}

val ints: Array<Int> = arrayOf(1, 2, 3)
val any = Array<Any>(3)
copy(ints, any) // 錯(cuò)誤: 期待的參數(shù)類(lèi)型是 (Array<Any>, Array<Any>)

我們需要確保的就是 copy() 函數(shù)不會(huì)做這類(lèi)不安全的操作. 我們希望禁止這個(gè)函數(shù)向 from 數(shù)組
寫(xiě)入 數(shù)據(jù), 我們可以這樣聲明:

fun copy(from: Array<out Any>, to: Array<Any>) {
  // ...
}

這種聲明在 Kotlin 中稱為 類(lèi)型投射(type projection): 我們聲明的含義是, from 不是一個(gè)單純的數(shù)組, 而是
一個(gè)被限制(投射)的數(shù)組: 我們只能對(duì)這個(gè)數(shù)組調(diào)用那些返回值為類(lèi)型參數(shù) T 的方法。

也可以使用 in 關(guān)鍵字來(lái)投射一個(gè)類(lèi)型。

fun fill(dest: Array<in String>, value: String) {
  // ...
}
  • 星號(hào)投射(Star-projection)
  • 泛型約束(Generic constraint)

對(duì)于一個(gè)給定的類(lèi)型參數(shù), 所允許使用的類(lèi)型, 可以通過(guò) 泛型約束(generic constraint) 來(lái)限制。

最常見(jiàn)的約束是 上界(upper bound), 與 Java 中的 extends 關(guān)鍵字相同:

fun <T : Comparable<T>> sort(list: List<T>) {
  // ...
}

對(duì)于類(lèi)型參數(shù) T , 只允許使用 Comparable<T> 的子類(lèi)型. 比如:

sort(listOf(1, 2, 3)) // 正確: Int 是 Comparable<Int> 的子類(lèi)型
sort(listOf(HashMap<Int, String>())) // 錯(cuò)誤: HashMap<Int, String> 不是
    // Comparable<HashMap<Int, String>> 的子類(lèi)型

泛型類(lèi)型

Java 里面的泛型不支持類(lèi)型, 比如 T.class這樣的代碼獲取不到類(lèi)型。Kotlin 泛型函數(shù)通過(guò)內(nèi)聯(lián)函數(shù)可以獲取泛型的類(lèi)型,比如:

inline fun <reified T>runtimeType(): Unit {
  println("My type parameter is " + T::class.qualifiedName)
}

inline fun <reified T>List<Any>.collect(): List<T> {
  return this.filter { it is T }.map { it as T }
}


感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

相關(guān)文章

  • idea注入mapper報(bào)錯(cuò)報(bào)紅的幾種解決方案

    idea注入mapper報(bào)錯(cuò)報(bào)紅的幾種解決方案

    相信大家在使用idea的時(shí)候一定會(huì)遇到這樣的問(wèn)題,就是在service里注入mapper的時(shí)候,明明代碼沒(méi)有問(wèn)題,也可以運(yùn)行,但是idea它就是給你報(bào)個(gè)錯(cuò),有個(gè)紅色的波浪線在下面,所以本文將給大家介紹了idea注入mapper報(bào)錯(cuò)報(bào)紅的幾種解決方案,需要的朋友可以參考下
    2023-12-12
  • java實(shí)現(xiàn)簡(jiǎn)單的圖書(shū)借閱系統(tǒng)

    java實(shí)現(xiàn)簡(jiǎn)單的圖書(shū)借閱系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)單的圖書(shū)借閱系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • Java8中Optional的一些常見(jiàn)錯(cuò)誤用法總結(jié)

    Java8中Optional的一些常見(jiàn)錯(cuò)誤用法總結(jié)

    我們知道 Java 8 增加了一些很有用的 API, 其中一個(gè)就是 Optional,下面這篇文章主要給大家介紹了關(guān)于Java8中Optional的一些常見(jiàn)錯(cuò)誤用法的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2018-07-07
  • SpringMVC+MyBatis 事務(wù)管理(實(shí)例)

    SpringMVC+MyBatis 事務(wù)管理(實(shí)例)

    本文先分析編程式注解事務(wù)和基于注解的聲明式事務(wù)。對(duì)SpringMVC+MyBatis 事務(wù)管理的相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧
    2017-08-08
  • Java Swing實(shí)現(xiàn)窗體添加背景圖片的2種方法詳解

    Java Swing實(shí)現(xiàn)窗體添加背景圖片的2種方法詳解

    這篇文章主要介紹了Java Swing實(shí)現(xiàn)窗體添加背景圖片的2種方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Swing實(shí)現(xiàn)窗體添加背景圖片的方法,并總結(jié)分析了Swing重繪中repaint與updateUI的區(qū)別,需要的朋友可以參考下
    2017-11-11
  • Java面試之高級(jí)特性基礎(chǔ)總結(jié)

    Java面試之高級(jí)特性基礎(chǔ)總結(jié)

    這篇文章主要為大家詳細(xì)介紹了10個(gè)Java高級(jí)特性基礎(chǔ)相關(guān)的問(wèn)題,也是大家面試中常常會(huì)遇到的問(wèn)題。文中的示例代講解詳細(xì),感興趣的可以了解一下
    2023-01-01
  • Java concurrency線程池之線程池原理(二)_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    Java concurrency線程池之線程池原理(二)_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    這篇文章主要為大家詳細(xì)介紹了Java concurrency線程池之線程池原理第二篇,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Spring?Boot?打包成Jar包運(yùn)行原理分析

    Spring?Boot?打包成Jar包運(yùn)行原理分析

    這篇文章主要為大家介紹了Spring?Boot?打包成Jar包運(yùn)行的原理分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • springboot中設(shè)置定時(shí)任務(wù)的三種方法小結(jié)

    springboot中設(shè)置定時(shí)任務(wù)的三種方法小結(jié)

    在我們開(kāi)發(fā)項(xiàng)目過(guò)程中,經(jīng)常需要定時(shí)任務(wù)來(lái)幫助我們來(lái)做一些內(nèi)容,本文介紹了springboot中設(shè)置定時(shí)任務(wù)的三種方法,主要包括@Scheduled注解,Quartz框架和xxl-job框架的實(shí)現(xiàn),感興趣的可以了解一下
    2023-12-12
  • Java 利用棧來(lái)反轉(zhuǎn)鏈表和排序的操作

    Java 利用棧來(lái)反轉(zhuǎn)鏈表和排序的操作

    這篇文章主要介紹了Java 利用棧來(lái)反轉(zhuǎn)鏈表和排序的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-02-02

最新評(píng)論