kotlin 泛型

fun <T> List<T>.slice(indices: IntRange): List<T>
<T>: 類(lèi)型函數(shù)聲明
List<T>: 接收者和返回類(lèi)型使用了類(lèi)型形參

(1)類(lèi)型參數(shù)約束
表示上界約束時(shí): fun <T: Number> List<T>.sum() :T
T : Number 中
T是類(lèi)型參數(shù)豫领,Number是上界

一旦使用了類(lèi)型形參T的上界魂角,你就可以把類(lèi)型T的值當(dāng)作它的上界(類(lèi)型)的值使用。
如:
// 指定Number為類(lèi)型形參的上界
fun <T:Number> oneHalf(value: T): Double{
// 調(diào)用Numbe類(lèi)中的方法
return value.toDouble()
}

(2)聲明帶類(lèi)型參數(shù)約束的函數(shù)
// 這個(gè)函數(shù)的實(shí)參必須是可比較的元素
fun <T: Comparable<T>> max(first: T, second: T): T{
return if (first > second) first else second
}
如果傳入無(wú)法比較的參數(shù)時(shí)伴网,代碼將無(wú)法編譯:
max("Kotlin", 42) // 拋出異常

(3)為一個(gè)類(lèi)型參數(shù)定義多個(gè)約束
where關(guān)鍵字 為一個(gè)類(lèi)型參數(shù)定義多個(gè)約束時(shí)使用
// 這里為類(lèi)型T指定多個(gè)約束 ,where 后就是約束
fun <T> ensureTrailingPeriod(seq: T) where T : CharSequence, T : Appendable{
// 調(diào)用 Appendable 接口的方法
if (!seq.endsWith('.')){
seq.append('.')
}
}

(4)讓類(lèi)型形參非空

// 這里聲明的泛型函數(shù)是沒(méi)有指定上界的盅粪,但是內(nèi)部機(jī)制會(huì)吧這個(gè)泛型
// 函數(shù)默認(rèn)使用 Any? 作為上界
class Processo<T> {
// 這里的 value 是可空的掷漱,盡管T沒(méi)有使用問(wèn)號(hào)標(biāo)記
fun process(value: T){
value?.hashCode()
}
}
fun testPro(){
// 可用類(lèi)型 String 用來(lái)替換T
val mullableStringProcesso = Processo<String?>()
// 使用 “null” 作為 “value”實(shí)參代碼可以編譯
mullableStringProcesso.process(null)
}

設(shè)置泛型始終都為不可空的
// 這里設(shè)置上界為 Amy ,確保了T類(lèi)型永遠(yuǎn)都是非空類(lèi)型
class Processo<T:Any> {
fun process(value: T){
value?.hashCode()
}
}
fun testPro(){
// 由于泛型的上界為 Any 不為空秦士,所以無(wú)法設(shè)置泛型類(lèi)型為可空類(lèi)型
val mullableStringProcesso = Processo<String?>()

}

(5)運(yùn)行時(shí)的泛型:擦出和實(shí)例化類(lèi)型參數(shù)
Jvm上的泛型一般是通過(guò)類(lèi)型擦除實(shí)現(xiàn)的。

(6)類(lèi)型的檢查和轉(zhuǎn)換
如何檢查一個(gè)值是否是列表:
如:
if (value is List<*>) { ... }
想要檢查一個(gè)值是否為列表栓辜,不可以在列表聲明使用具體類(lèi)型的列表恋拍,必須使用 * 來(lái)代替泛型類(lèi)型才可以對(duì)這個(gè)值進(jìn)行檢查
如:
if( value is List<String>) { ... } // 這里使用了具體的 String,所以代碼不會(huì)編譯

" * " : 稱為星號(hào)投影

(7)對(duì)未知泛型類(lèi)型 和 已知類(lèi)型做類(lèi)型轉(zhuǎn)換
如:
// 這里 c 的泛型類(lèi)型 是未知的
fun printlnSum( c: Collection<*> ){
val intList = c as? List<Int> ? :throw IllegalArgumentException("List is expected")
println( inList.sum() )
}
調(diào)用:
println( setOf(1, 2, 3)) // Set不是列表,拋出異常
println(listOf("a", "b", "c" )) // 類(lèi)型轉(zhuǎn)換成功藕甩,但后面拋出了另外的異常

如果吧 * 號(hào)換成具體類(lèi)型施敢,那么轉(zhuǎn)換就可以成功

(8)聲明帶實(shí)化類(lèi)型參數(shù)的函數(shù)
fun <T> isA(value : Any) = value is T
// 這段代碼是無(wú)法編譯的周荐,因?yàn)闊o(wú)法知道泛型的類(lèi)型

reified: reified 的意思是實(shí)化(具體化)。而作為 Kotlin 的一個(gè)方法 泛型 關(guān)鍵字僵娃,它代表你可以在方法體內(nèi)訪問(wèn)泛型指定的JVM類(lèi)對(duì)象概作。必須以 inline 內(nèi)聯(lián)方式聲明這個(gè)方法才有效。

理解: 內(nèi)聯(lián)函數(shù)的類(lèi)型實(shí)參能夠被實(shí)化默怨,意味著你可以在運(yùn)行時(shí)引用實(shí)際的類(lèi)型實(shí)參讯榕,reified 關(guān)鍵字的聲明就是讓當(dāng)前泛型類(lèi)型可以在運(yùn)行時(shí)得到具體的類(lèi)型,就是類(lèi)型參數(shù)不會(huì)在運(yùn)行時(shí)擦除匙睹。

加上inline reified 后愚屁,函數(shù)可以編譯
// 聲明帶實(shí)體化類(lèi)型參數(shù)的函數(shù)
// reified T : 可以在運(yùn)行時(shí)得到這個(gè)泛型的類(lèi)型
inline fun <reified T> isA(value: Any) = value is T

(9)使用標(biāo)準(zhǔn)庫(kù)的函數(shù)

val items = listOf("one" ,2 ,"three")
/*
使用它來(lái)檢查列表中的值是不是指定為該類(lèi)型實(shí)參的類(lèi)實(shí)例

println(items.filterIsInstance<String>())
[one, three]
*/

// "reified" 聲明了類(lèi)型參數(shù)不會(huì)在運(yùn)行時(shí)擦除
inline fun <reified T> Iterable<*>.filterIsInstance() : List<T>{
val destination = mutableListOf<T>()
// 迭代當(dāng)前列表實(shí)例中的元素并放入element
for (element in this) {
// 可以檢查元素是不是為指定為類(lèi)型實(shí)參類(lèi)的實(shí)例
if (element is T){
destination.add(element)
}
}
return destination
}

為什么實(shí)化只對(duì)內(nèi)聯(lián)函數(shù)有效:
編譯把實(shí)現(xiàn)內(nèi)聯(lián)函數(shù)的字節(jié)碼插入每一次調(diào)用發(fā)生的地方,每次你調(diào)用帶實(shí)化類(lèi)型參數(shù)的函數(shù)時(shí)痕檬,編譯器都知道這次特定調(diào)用中用作列席實(shí)參的確切類(lèi)型霎槐。因此,編譯器都知道這次特定調(diào)用中作為類(lèi)型實(shí)參的具體字節(jié)碼梦谜。
如:
filterIsInstance<String>

for (element in this){
if(element is String){
destination.add(element)
}
}
著兩段代碼時(shí)等價(jià)的

(10)調(diào)用實(shí)化類(lèi)型參數(shù)代替類(lèi)引用
val serviceImpl = ServiceLoader.load(Service::class.java)
:: class.java 的語(yǔ)法展現(xiàn)了如何獲取java.lang.Class 對(duì)應(yīng)的Kotlin類(lèi)

使用帶實(shí)化參數(shù)的函數(shù)重寫(xiě)這個(gè)例子
val serviceImpl = loadService<Service>()

具體實(shí)現(xiàn)
// 類(lèi)型參數(shù)標(biāo)記成了 “reified”
inline fun <reified T: Activity> Context.startActivity(){
// 把T::class 當(dāng)成類(lèi)型參數(shù)的類(lèi)訪問(wèn)
val intent = Intent(this, T::class.java)
startActivity(intent)
}

oninline: 這個(gè)標(biāo)記關(guān)鍵字可以把內(nèi)聯(lián)函數(shù)的某段語(yǔ)句或者字段標(biāo)記為非內(nèi)聯(lián)的

(11)類(lèi)丘跌,類(lèi)型和子類(lèi)型;

子類(lèi):如B 實(shí)現(xiàn)了 A 就等于B 是A 的子類(lèi)

類(lèi)型:兩種對(duì)象的分類(lèi)唁桩,如Int 是 Number類(lèi)型碍岔,反過(guò)來(lái)Number是Int類(lèi)型

子類(lèi)型:如:Int? 可以存入 整型數(shù)據(jù)和null , int 只可以存入整型數(shù)據(jù)朵夏,所以Int 是 Int? 的子類(lèi)型

(12)協(xié)變
如果A是B的子類(lèi)型,那么List<A> 就是 List<B>的子類(lèi)型榆纽,這樣的類(lèi)或者接口被稱為協(xié)變的仰猖。

函數(shù)的參數(shù)類(lèi)型叫做in位置,函數(shù)的返回類(lèi)型叫做out位置奈籽。

協(xié)變:
open class Animal{}
class Herd<T: Animal>{}
fun feedAll(animals: Herd<Animal>){}

class Cat: Animal(){}
fun takeCareOfCats(cats: Herd<cat>){
// 這里即使cats 的類(lèi)型是Herd<Cat> 類(lèi)型的饥侵,而Cat又是繼承Animal類(lèi),也無(wú)法編譯
feedAll(cats)
}

如果把 Herd<T: Animal> 的T前面加上 out 代碼就可以編譯成功

重申:(關(guān)鍵字 out)
Herd<out T: Animal >時(shí)衣屏,子類(lèi)型化會(huì)被保留(producer<Cat>)是Producer<Animal>的子類(lèi)型

// MutableList不能在T上聲明成協(xié)變的
interface MuableList<T>:List<T>, MutableCollection<T>{
// 因?yàn)門(mén)用在了in位置
override fun add(element: T): Boolean
}

(13)逆變:反轉(zhuǎn)子類(lèi)型化關(guān)系(關(guān)鍵字 in)
逆變: 如果B 是A 的子類(lèi)型躏升,那么Comsumer<A>就是 Comsumer<B>的子類(lèi)型。狼忱,類(lèi)型A和類(lèi)型B交換了位置膨疏,所以我們說(shuō)子類(lèi)型被反轉(zhuǎn)了。

協(xié)變钻弄,逆變總結(jié):fun transform(t: T): T ,t : T 這個(gè)T是in位置佃却,返回值類(lèi)型的T是out位置

Kotlin的
MutableList<out T> 和java 中的 MutableList< ? extends T> 是一個(gè)意思
MutableList<in T> 和Java 中的MutableList<? super T> 是一個(gè)意思

協(xié)變:
producer<out T>
類(lèi)的子類(lèi)型化保留了:producer<Car>是Producer<Animal>的子類(lèi)型(可以使用T的子類(lèi)型)
T只能在out位置
逆變:
Consumer<in: T>
子類(lèi)型化反轉(zhuǎn)了: Comsumer<Animal>是Consumer<Cat>的類(lèi)型(可以使用T的父類(lèi)型)
T只能在in位置
不變型:
MutableList<T>
沒(méi)有子類(lèi)型化
T可以在如何位置

使用點(diǎn)變型:在類(lèi)型出現(xiàn)的地方指定變型
使用不變型類(lèi)型參數(shù)的數(shù)據(jù)拷貝函數(shù) ,能夠把一個(gè)列表中的元素拷貝到拎一個(gè)列表
引入一個(gè)泛型:
fun <T> copyData(source: MutableList<T>), destination: MutableList<T>{
for(Item in source){
destination.add(item)
}
}

引入多個(gè)泛型:
fun <T: R, R> copyData(source: MutableList<T>, destination: MutableList<R>){
for(Item in source){
destination.add(item)
}
}

帶out投影類(lèi)型的數(shù)據(jù)拷貝函數(shù):
// 可以給類(lèi)型的用法加上out 關(guān)鍵字,沒(méi)有使用那些T 用在 in 位置的方法
fun <T> copyData(source: MuableList<out T> , destination: MutableList<T>){
for(item in source){
destination.add(item)
}
}

帶in投影類(lèi)型參數(shù)的數(shù)據(jù)拷貝函數(shù)
// 允許目標(biāo)元素的類(lèi)型是來(lái)源元素類(lèi)型的超類(lèi)型
fun <T> copyData(source: MutableList<T> , destination: MutableList<in T>){
for(item in source){
destination.add(item)
}
}

(14)星號(hào)投影
寫(xiě)法一:List<*> 一個(gè)包含未知類(lèi)型元素的列表窘俺,相當(dāng)于Java的List<?>

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末饲帅,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌灶泵,老刑警劉巖育八,帶你破解...
    沈念sama閱讀 222,104評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異赦邻,居然都是意外死亡髓棋,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)深纲,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)仲锄,“玉大人,你說(shuō)我怎么就攤上這事湃鹊∪搴埃” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,697評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵币呵,是天一觀的道長(zhǎng)怀愧。 經(jīng)常有香客問(wèn)我,道長(zhǎng)余赢,這世上最難降的妖魔是什么芯义? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,836評(píng)論 1 298
  • 正文 為了忘掉前任,我火速辦了婚禮妻柒,結(jié)果婚禮上扛拨,老公的妹妹穿的比我還像新娘。我一直安慰自己举塔,他們只是感情好绑警,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著央渣,像睡著了一般计盒。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上芽丹,一...
    開(kāi)封第一講書(shū)人閱讀 52,441評(píng)論 1 310
  • 那天北启,我揣著相機(jī)與錄音,去河邊找鬼拔第。 笑死咕村,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的蚊俺。 我是一名探鬼主播培廓,決...
    沈念sama閱讀 40,992評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼春叫!你這毒婦竟也來(lái)了肩钠?” 一聲冷哼從身側(cè)響起泣港,我...
    開(kāi)封第一講書(shū)人閱讀 39,899評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎价匠,沒(méi)想到半個(gè)月后当纱,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,457評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡踩窖,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評(píng)論 3 341
  • 正文 我和宋清朗相戀三年坡氯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片洋腮。...
    茶點(diǎn)故事閱讀 40,664評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡箫柳,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出啥供,到底是詐尸還是另有隱情悯恍,我是刑警寧澤,帶...
    沈念sama閱讀 36,346評(píng)論 5 350
  • 正文 年R本政府宣布伙狐,位于F島的核電站涮毫,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏贷屎。R本人自食惡果不足惜罢防,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望唉侄。 院中可真熱鬧咒吐,春花似錦、人聲如沸属划。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,511評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)榴嗅。三九已至,卻和暖如春陶舞,著一層夾襖步出監(jiān)牢的瞬間嗽测,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,611評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工肿孵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留唠粥,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,081評(píng)論 3 377
  • 正文 我出身青樓停做,卻偏偏與公主長(zhǎng)得像晤愧,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子蛉腌,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • 本博文主要講解一些Kotlin泛型的問(wèn)題官份,中間會(huì)對(duì)比穿插Java泛型只厘。 1. 泛型類(lèi)型參數(shù) 1.1 形式 我們使用...
    24K男閱讀 18,445評(píng)論 4 11
  • 建議先閱讀我的上一篇文章 -- Java 泛型 和 Java 泛型一樣,Kotlin 泛型也是 Kotlin 語(yǔ)言...
    JohnnyShieh閱讀 6,504評(píng)論 1 26
  • 寫(xiě)在開(kāi)頭:本人打算開(kāi)始寫(xiě)一個(gè)Kotlin系列的教程舅巷,一是使自己記憶和理解的更加深刻羔味,二是可以分享給同樣想學(xué)習(xí)Kot...
    胡奚冰閱讀 1,436評(píng)論 1 3
  • 轉(zhuǎn)載文章,出處: https://blog.kotliner.cn/2017/06/26/kotlin-gener...
    _10_01_閱讀 693評(píng)論 0 0
  • 泛型 泛型(Generic Type)簡(jiǎn)介 通常情況的類(lèi)和函數(shù),我們只需要使用具體的類(lèi)型即可:要么是基本類(lèi)型飒房,要么...
    Tenderness4閱讀 1,421評(píng)論 4 2