Kotlin 泛型詳解及簡(jiǎn)單實(shí)例
Kotlin 泛型詳解
概述
一般類和函數(shù),只能使用具體的類型:要么是基本類型,要么是自定義的類。如果要編寫(xiě)可以應(yīng)用于多種類型的代碼,這種刻板的約束對(duì)代碼的限制很大。而OOP的多態(tài)采用了一種泛化的機(jī)制,在SE 5種,Java引用了泛型。泛型,即“參數(shù)化類型”。一提到參數(shù),最熟悉的就是定義方法時(shí)有形參,然后調(diào)用此方法時(shí)傳遞實(shí)參。那么參數(shù)化類型怎么理解呢?顧名思義,就是將類型由原來(lái)的具體的類型參數(shù)化,類似于方法中的變量參數(shù),此時(shí)類型也定義成參數(shù)形式(可以稱之為類型形參),然后在使用/調(diào)用時(shí)傳入具體的類型(類型實(shí)參)。
在Kotlin中,依然可以使用泛型,解耦類與函數(shù)與所用類型之間的約束,甚至是使用方法都與Java一致。
泛型類
聲明一個(gè)泛型類
class Box<T>(t: T) { var value = t }
通常, 要?jiǎng)?chuàng)建這樣一個(gè)類的實(shí)例, 我們需要指定類型參數(shù):
val box: Box<Int> = Box<Int>(1)
但是, 如果類型參數(shù)可以通過(guò)推斷得到, 比如, 通過(guò)構(gòu)造器參數(shù)類型, 或通過(guò)其他手段推斷得到, 此時(shí)允許省略類型參數(shù):
val box = Box(1) // 1 的類型為 Int, 因此編譯器知道我們創(chuàng)建的實(shí)例是 Box<Int> 類型
泛型函數(shù)
泛型函數(shù)與其所在的類是否是泛型沒(méi)有關(guān)系。泛型函數(shù)使得該函數(shù)能夠獨(dú)立于其所在類而產(chǎn)生變化。在<Thinking in Java>有這么一句話:無(wú)論何時(shí)只要你能做到,你就應(yīng)該盡量使用泛型方法,也就是說(shuō)如果使用泛型方法可以取代將整個(gè)類泛型化,那么就應(yīng)該只使用泛型方法,因?yàn)樗梢允故虑楦靼?。這種泛型使用思想,在Kotlin中依然可以延續(xù)。
下面我們聲明了一個(gè)泛型函數(shù)doPrintln,當(dāng)T是一個(gè)Int類型時(shí),打印其個(gè)位的值;如果T是String類型,將字母全部大寫(xiě)輸出;如果是其他類型,打印“T is not Int and String”。
fun main(args: Array<String>) { val age = 23 val name = "Jone" val person = true doPrintln(age) // 打?。? doPrintln(name) // 打?。篔ONE 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") } }
注:
- 類型參數(shù)放在函數(shù)名稱之前。
- 如果在調(diào)用處明確地傳入了類型參數(shù), 那么類型參數(shù)應(yīng)該放在函數(shù)名稱 之后。如果不傳入?yún)?shù)類型,編譯器會(huì)根據(jù)傳入的值自動(dòng)推斷參數(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) // 打?。篶lass com.teaphy.generic.Box println(boxString.javaClass) // 打?。篶lass com.teaphy.generic.Box }
現(xiàn)聲明了一個(gè)泛型類Box<T>,在不同的類型的類型在行為方面肯定不一樣,但是在我們獲取其所在類時(shí),我們只是得到了“class com.teaphy.generic.Box”。在這里我們不得不面對(duì)一個(gè)殘酷的現(xiàn)實(shí):在泛型內(nèi)部,無(wú)法獲得任何有關(guān)泛型參數(shù)類型的信息。
不管是Java還是Kotlin,泛型都是使用擦除來(lái)實(shí)現(xiàn)的,這意味著當(dāng)你在使用泛型時(shí),任務(wù)具體的類型信息都被擦除的,你唯一知道的就是你再使用一個(gè)對(duì)象。比如,Box<String>和Box<Int>在運(yùn)行時(shí)是想的類型,都是Box的實(shí)例。在使用泛型時(shí),具體類型信息的擦除是我們不不懂得不面對(duì)的,在Kotlin中也為我們提供了一些可供參考的解決方案:
- 類型協(xié)變
- 類型投射
- 泛型約束
類型協(xié)變
在類型聲明時(shí),使用協(xié)變注解修飾符(in或者out)。于這個(gè)注解出現(xiàn)在類型參數(shù)的聲明處, 因此我們稱之為聲明處的類型變異。如果在使用泛型時(shí),使用了該類型編譯了會(huì)有什么效果呢?
假設(shè)我們有一個(gè)泛型接口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: 來(lái)確保Source的成員函數(shù)只能消費(fèi)T類型,而不能返回T類型
- out R:來(lái)確保Source的成員函數(shù)只能返回R類型,而不能消費(fèi)R類型
從上面的解釋中,我們可以清楚的知道了協(xié)變注解in和out的用意,其實(shí)際上是定義了類型參數(shù)在該類或者接口的用途,是用來(lái)消費(fèi)的還是用來(lái)返回的,對(duì)其做了相應(yīng)的限定。
類型投射
上面我們已經(jīng)了解到了協(xié)變注解in和out的用意,下面我們將會(huì)用in和out,做一件有意義的事,看下面代碼
fun copy(from: Array<out String>, to: Array<Any>) { // ... } fun fill(dest: Array<in String>, value: String) { // ... }
對(duì)于copy函數(shù)中中,from的泛型參數(shù)使用了協(xié)變注解out修飾,意味著該參數(shù)不能在該函數(shù)中消費(fèi),也就是說(shuō)在該函數(shù)中禁止對(duì)該參數(shù)進(jìn)行任何操作。
對(duì)于fill函數(shù)中,dest的泛型參數(shù)使用了協(xié)變注解in修飾,Array<in String>與Java的 Array<? super String> 相同, 也就是說(shuō), 你可以使用CharSequence數(shù)組,或者 Object 數(shù)組作為 fill() 函數(shù)的參數(shù)
這種聲明在Kotlin中稱為類型投射(type projection),類型投射的主要用于對(duì)參數(shù)做了相對(duì)因的限定,避免了對(duì)該參數(shù)類的不安全操作。
星號(hào)投射
有些時(shí)候, 你可能想表示你并不知道類型參數(shù)的任何信息, 但是仍然希望能夠安全地使用它. 這里所謂”安全地使用”是指, 對(duì)泛型類型定義一個(gè)類型投射, 要求這個(gè)泛型類型的所有的實(shí)體實(shí)例, 都是這個(gè)投射的子類型.
對(duì)于這個(gè)問(wèn)題, Kotlin 提供了一種語(yǔ)法, 稱為 星號(hào)投射(star-projection):
- 假如類型定義為 Foo<out T> , 其中 T 是一個(gè)協(xié)變的類型參數(shù), 上界(upper bound)為 TUpper ,Foo<> 等價(jià)于 Foo<out TUpper> . 它表示, 當(dāng) T 未知時(shí), 你可以安全地從 Foo<> 中 讀取TUpper 類型的值.
- 假如類型定義為 Foo<in T> , 其中 T 是一個(gè)反向協(xié)變的類型參數(shù), Foo<> 等價(jià)于 Foo<inNothing> . 它表示, 當(dāng) T 未知時(shí), 你不能安全地向 Foo<> 寫(xiě)入 任何東西.
- 假如類型定義為 Foo<T> , 其中 T 是一個(gè)協(xié)變的類型參數(shù), 上界(upper bound)為 TUpper , 對(duì)于讀取值的場(chǎng)合, Foo<*> 等價(jià)于 Foo<out TUpper> , 對(duì)于寫(xiě)入值的場(chǎng)合, 等價(jià)于 Foo<in Nothing> .
如果一個(gè)泛型類型中存在多個(gè)類型參數(shù), 那么每個(gè)類型參數(shù)都可以單獨(dú)的投射. 比如, 如果類型定義為interface Function<in T, out U> , 那么可以出現(xiàn)以下幾種星號(hào)投射:
- Function<*, String> , 代表 Function<in Nothing, String> ;
- Function<Int, *> , 代表 Function<Int, out Any?> ;
- Function<, > , 代表 Function<in Nothing, out Any?> .
注意: 星號(hào)投射與 Java 的原生類型(raw type)非常類似, 但可以安全使用
泛型約束
對(duì)于一個(gè)給定的類型參數(shù), 所允許使用的類型, 可以通過(guò)泛型約束(generic constraint) 來(lái)限制。
上界
最常見(jiàn)的約束是 上界(upper bound):
fun <T : Comparable<T>> sort(list: List<T>) { // ... }
冒號(hào)之后指定的類型就是類型參數(shù)的 上界(upper bound): 對(duì)于類型參數(shù) T , 只允許使用 Comparable<T>的子類型. 比如:
sort(listOf(1, 2, 3)) // 正確: Int 是 Comparable<Int> 的子類型 sort(listOf(HashMap<Int, String>())) // 錯(cuò)誤: HashMap<Int, String> 不是 Comparable<HashMap<Int, String>> 的子類型
如果沒(méi)有指定, 則默認(rèn)使用的上界是 Any? . 在定義類型參數(shù)的尖括號(hào)內(nèi), 只允許定義唯一一個(gè)上界. 如果同一個(gè)類型參數(shù)需要指定多個(gè)上界, 這時(shí)就需要使用單獨(dú)的 where 子句:
fun <T> cloneWhenGreater(list: List<T>, threshold: T): List<T> where T : Comparable, T : Cloneable { return list.filter { it > threshold }.map { it.clone() } }
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
MybatisPlus多數(shù)據(jù)源及事務(wù)解決思路
這篇文章主要介紹了MybatisPlus多數(shù)據(jù)源及事務(wù)解決思路,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01Eclipse啟動(dòng)Tomcat超時(shí)問(wèn)題的解決方法
2013-03-03Java如何跳出當(dāng)前多重循環(huán)你知道嗎
這篇文章主要為大家介紹了Java跳出當(dāng)前多重循環(huán),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-01-01SSH框架網(wǎng)上商城項(xiàng)目第1戰(zhàn)之整合Struts2、Hibernate4.3和Spring4.2
SSH框架網(wǎng)上商城項(xiàng)目第1戰(zhàn)之整合Struts2、Hibernate4.3和Spring4.2,感興趣的小伙伴們可以參考一下2016-05-05Hibernate中l(wèi)oad方法與get方法的區(qū)別
Hibernate中有兩個(gè)極為相似的方法get()與load(),他們都可以通過(guò)指定的實(shí)體類與ID從數(shù)據(jù)庫(kù)中讀取數(shù)據(jù),并返回對(duì)應(yīng)的實(shí)例,但Hibernate不會(huì)搞兩個(gè)完全一樣的方法的2016-01-01SpringBoot實(shí)現(xiàn)抽獎(jiǎng)算法的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何通過(guò)SpringBoot實(shí)現(xiàn)抽獎(jiǎng)算法,文中的示例代碼簡(jiǎn)潔易懂,具有一定的參考價(jià)值,感興趣的小伙伴可以了解一下2023-06-06