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

Kotlin 泛型詳解及簡單實例

 更新時間:2017年06月05日 10:33:19   作者:行云間  
這篇文章主要介紹了 Kotlin 泛型詳解及簡單實例的相關資料,需要的朋友可以參考下

 Kotlin 泛型詳解

概述

一般類和函數(shù),只能使用具體的類型:要么是基本類型,要么是自定義的類。如果要編寫可以應用于多種類型的代碼,這種刻板的約束對代碼的限制很大。而OOP的多態(tài)采用了一種泛化的機制,在SE 5種,Java引用了泛型。泛型,即“參數(shù)化類型”。一提到參數(shù),最熟悉的就是定義方法時有形參,然后調用此方法時傳遞實參。那么參數(shù)化類型怎么理解呢?顧名思義,就是將類型由原來的具體的類型參數(shù)化,類似于方法中的變量參數(shù),此時類型也定義成參數(shù)形式(可以稱之為類型形參),然后在使用/調用時傳入具體的類型(類型實參)。

在Kotlin中,依然可以使用泛型,解耦類與函數(shù)與所用類型之間的約束,甚至是使用方法都與Java一致。

泛型類

聲明一個泛型類

class Box<T>(t: T) {
  var value = t
}

通常, 要創(chuàng)建這樣一個類的實例, 我們需要指定類型參數(shù):

val box: Box<Int> = Box<Int>(1)

但是, 如果類型參數(shù)可以通過推斷得到, 比如, 通過構造器參數(shù)類型, 或通過其他手段推斷得到, 此時允許省略類型參數(shù):

val box = Box(1) // 1 的類型為 Int, 因此編譯器知道我們創(chuàng)建的實例是 Box&lt;Int> 類型

泛型函數(shù)

泛型函數(shù)與其所在的類是否是泛型沒有關系。泛型函數(shù)使得該函數(shù)能夠獨立于其所在類而產(chǎn)生變化。在<Thinking in Java>有這么一句話:無論何時只要你能做到,你就應該盡量使用泛型方法,也就是說如果使用泛型方法可以取代將整個類泛型化,那么就應該只使用泛型方法,因為它可以使事情更明白。這種泛型使用思想,在Kotlin中依然可以延續(xù)。

下面我們聲明了一個泛型函數(shù)doPrintln,當T是一個Int類型時,打印其個位的值;如果T是String類型,將字母全部大寫輸出;如果是其他類型,打印“T is not Int and String”。

fun main(args: Array<String>) {
  val age = 23
  val name = "Jone"
  val person = true

  doPrintln(age) // 打?。?
  doPrintln(name) // 打印:JONE
  doPrintln(person) // 打?。篢 is not Int and String
}

fun <T> doPrintln(content: T) {

  when (content) {
    is Int -> println(content % 10) 
    is String -> println(content.toUpperCase())
    else -> println("T is not Int and String")
  }
}

注:

  1. 類型參數(shù)放在函數(shù)名稱之前。
  2. 如果在調用處明確地傳入了類型參數(shù), 那么類型參數(shù)應該放在函數(shù)名稱 之后。如果不傳入?yún)?shù)類型,編譯器會根據(jù)傳入的值自動推斷參數(shù)類型。

擦除的神秘之處

下面我們先看一段代碼:

class Box<T>(t : T) {
  var value = t
}


fun main(args: Array<String>) {
  var boxInt = Box<Int>(10)
  var boxString = Box<String>("Jone")

  println(boxInt.javaClass) // 打印:class com.teaphy.generic.Box
  println(boxString.javaClass) // 打?。篶lass com.teaphy.generic.Box
}

現(xiàn)聲明了一個泛型類Box<T>,在不同的類型的類型在行為方面肯定不一樣,但是在我們獲取其所在類時,我們只是得到了“class com.teaphy.generic.Box”。在這里我們不得不面對一個殘酷的現(xiàn)實:在泛型內(nèi)部,無法獲得任何有關泛型參數(shù)類型的信息。

不管是Java還是Kotlin,泛型都是使用擦除來實現(xiàn)的,這意味著當你在使用泛型時,任務具體的類型信息都被擦除的,你唯一知道的就是你再使用一個對象。比如,Box<String>和Box<Int>在運行時是想的類型,都是Box的實例。在使用泛型時,具體類型信息的擦除是我們不不懂得不面對的,在Kotlin中也為我們提供了一些可供參考的解決方案:

  • 類型協(xié)變
  • 類型投射
  • 泛型約束

類型協(xié)變

在類型聲明時,使用協(xié)變注解修飾符(in或者out)。于這個注解出現(xiàn)在類型參數(shù)的聲明處, 因此我們稱之為聲明處的類型變異。如果在使用泛型時,使用了該類型編譯了會有什么效果呢?

假設我們有一個泛型接口Source<in T, out R>, 其中T由協(xié)變注解in修飾,R由協(xié)變注解Out修飾.

internal interface Source<in T, out R> {
  fun mapT(t: T): Unit
  fun nextR(): R
}
  • in T: 來確保Source的成員函數(shù)只能消費T類型,而不能返回T類型
  • out R:來確保Source的成員函數(shù)只能返回R類型,而不能消費R類型

從上面的解釋中,我們可以清楚的知道了協(xié)變注解in和out的用意,其實際上是定義了類型參數(shù)在該類或者接口的用途,是用來消費的還是用來返回的,對其做了相應的限定。

類型投射

上面我們已經(jīng)了解到了協(xié)變注解in和out的用意,下面我們將會用in和out,做一件有意義的事,看下面代碼

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

fun fill(dest: Array<in String>, value: String) {
  // ...
}

對于copy函數(shù)中中,from的泛型參數(shù)使用了協(xié)變注解out修飾,意味著該參數(shù)不能在該函數(shù)中消費,也就是說在該函數(shù)中禁止對該參數(shù)進行任何操作。

對于fill函數(shù)中,dest的泛型參數(shù)使用了協(xié)變注解in修飾,Array<in String>與Java的 Array<? super String> 相同, 也就是說, 你可以使用CharSequence數(shù)組,或者 Object 數(shù)組作為 fill() 函數(shù)的參數(shù)

這種聲明在Kotlin中稱為類型投射(type projection),類型投射的主要用于對參數(shù)做了相對因的限定,避免了對該參數(shù)類的不安全操作。

星號投射

有些時候, 你可能想表示你并不知道類型參數(shù)的任何信息, 但是仍然希望能夠安全地使用它. 這里所謂”安全地使用”是指, 對泛型類型定義一個類型投射, 要求這個泛型類型的所有的實體實例, 都是這個投射的子類型.

對于這個問題, Kotlin 提供了一種語法, 稱為 星號投射(star-projection):

  1. 假如類型定義為 Foo<out T> , 其中 T 是一個協(xié)變的類型參數(shù), 上界(upper bound)為 TUpper ,Foo<> 等價于 Foo<out TUpper> . 它表示, 當 T 未知時, 你可以安全地從 Foo<> 中 讀取TUpper 類型的值.
  2. 假如類型定義為 Foo<in T> , 其中 T 是一個反向協(xié)變的類型參數(shù), Foo<> 等價于 Foo<inNothing> . 它表示, 當 T 未知時, 你不能安全地向 Foo<> 寫入 任何東西.
  3. 假如類型定義為 Foo<T> , 其中 T 是一個協(xié)變的類型參數(shù), 上界(upper bound)為 TUpper , 對于讀取值的場合, Foo<*> 等價于 Foo<out TUpper> , 對于寫入值的場合, 等價于 Foo<in Nothing> .

如果一個泛型類型中存在多個類型參數(shù), 那么每個類型參數(shù)都可以單獨的投射. 比如, 如果類型定義為interface Function<in T, out U> , 那么可以出現(xiàn)以下幾種星號投射:

  1. Function<*, String> , 代表 Function<in Nothing, String> ;
  2. Function<Int, *> , 代表 Function<Int, out Any?> ;
  3. Function<, > , 代表 Function<in Nothing, out Any?> .

注意: 星號投射與 Java 的原生類型(raw type)非常類似, 但可以安全使用

泛型約束

對于一個給定的類型參數(shù), 所允許使用的類型, 可以通過泛型約束(generic constraint) 來限制。

上界

最常見的約束是 上界(upper bound):

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

冒號之后指定的類型就是類型參數(shù)的 上界(upper bound): 對于類型參數(shù) T , 只允許使用 Comparable<T>的子類型. 比如:

sort(listOf(1, 2, 3)) // 正確: Int 是 Comparable&lt;Int> 的子類型
sort(listOf(HashMap<Int, String>())) // 錯誤: HashMap<Int, String> 不是
Comparable<HashMap<Int, String>> 的子類型

如果沒有指定, 則默認使用的上界是 Any? . 在定義類型參數(shù)的尖括號內(nèi), 只允許定義唯一一個上界. 如果同一個類型參數(shù)需要指定多個上界, 這時就需要使用單獨的 where 子句:

fun <T> cloneWhenGreater(list: List<T>, threshold: T): List<T> where T : Comparable,
T : Cloneable {
  return list.filter { it > threshold }.map { it.clone() }
}

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

相關文章

  • MybatisPlus多數(shù)據(jù)源及事務解決思路

    MybatisPlus多數(shù)據(jù)源及事務解決思路

    這篇文章主要介紹了MybatisPlus多數(shù)據(jù)源及事務解決思路,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-01-01
  • Java如何跳出當前多重循環(huán)你知道嗎

    Java如何跳出當前多重循環(huán)你知道嗎

    這篇文章主要為大家介紹了Java跳出當前多重循環(huán),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • SSH框架網(wǎng)上商城項目第1戰(zhàn)之整合Struts2、Hibernate4.3和Spring4.2

    SSH框架網(wǎng)上商城項目第1戰(zhàn)之整合Struts2、Hibernate4.3和Spring4.2

    SSH框架網(wǎng)上商城項目第1戰(zhàn)之整合Struts2、Hibernate4.3和Spring4.2,感興趣的小伙伴們可以參考一下
    2016-05-05
  • Hibernate中l(wèi)oad方法與get方法的區(qū)別

    Hibernate中l(wèi)oad方法與get方法的區(qū)別

    Hibernate中有兩個極為相似的方法get()與load(),他們都可以通過指定的實體類與ID從數(shù)據(jù)庫中讀取數(shù)據(jù),并返回對應的實例,但Hibernate不會搞兩個完全一樣的方法的
    2016-01-01
  • Java中的自旋鎖解析

    Java中的自旋鎖解析

    這篇文章主要介紹了Java中的自旋鎖解析,自旋鎖是指當一個線程嘗試獲取某個鎖時,如果該鎖已被其他線程占用,就一直循環(huán)檢測鎖是否被釋放,而不是進入線程掛起或睡眠狀態(tài),需要的朋友可以參考下
    2023-10-10
  • 寶塔升級JDK版本超詳細圖文教程

    寶塔升級JDK版本超詳細圖文教程

    寶塔自動安裝的JDK是一種用于開發(fā)和運行Java程序的軟件開發(fā)工具包,下面這篇文章主要給大家介紹了關于寶塔升級JDK版本的相關資料,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-12-12
  • SpringBoot實現(xiàn)抽獎算法的示例代碼

    SpringBoot實現(xiàn)抽獎算法的示例代碼

    這篇文章主要為大家詳細介紹了如何通過SpringBoot實現(xiàn)抽獎算法,文中的示例代碼簡潔易懂,具有一定的參考價值,感興趣的小伙伴可以了解一下
    2023-06-06
  • eclipse安裝spring ide的步驟詳解

    eclipse安裝spring ide的步驟詳解

    這篇文章主要介紹了eclipse安裝spring ide的步驟詳解,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-10-10
  • 一文詳解Spring?AOP的配置與使用

    一文詳解Spring?AOP的配置與使用

    面向切面編程(俗稱AOP)提供了一種面向對象編程(俗稱OOP)的補充,面向對象編程最核心的單元是類(class),然而面向切面編程最核心的單元是切面(Aspects)。本文就來和大家聊聊AOP的配置與使用,感興趣的可以了解一下
    2022-11-11

最新評論