Kotlin学习笔记之 10 泛型

首发于公众号: DSGtalk1989

10.Kotlin 泛型

  • 泛型约束

    就是我们在java中看到的T extends User,意思就是谁接受的泛型只能是User或者User的子类,在kotlin中我们使用:来代替

    fun <T : Number> List<T>.sum():T{
          //...
      }
    

    这里直接将放行和拓展函数连在了一块儿。

    kotlin厉害的地方是提供了多个上界,比如说,我现在需要T既实现了接口A,也实现了接口B,我们无法直接通过T : A的方式来进行处理,可以使用where关键字

    fun <T> List<T>.sum():T where T : A, T : B{
          //...
      }
    

    这个T需要同时实现AB

  • 型变in和out

    我们先来了解java中的两个概念,协变和逆变。

    List<A> aListList<B> bListA extends B。我们可以bList.add(new A()) 但是我们无法aList.add(new B()),但是,bList不能addAll``aListList<A>不是List<B>的子类,这个大家应该都能理解。

    有了这一层引导基础,我们结合泛型再来看一下。

    List<? extends B> list
    
    list.add(new A())  OK
    list.addAll(List<A>)  not OK
    

    我们一般把? extends称之为协变。同样的我们也能去理解List<? super B>,最低我们只能add B,只能往上不断的往里面扔B的父类,B的子类无法进入到这个list中。我们将? super称之为逆变。

    结合inout,在kotlin中我们怎么去运用这两个操作符,并且跟协变和逆变结合起来。

    比较容易的记忆方法是,如果我们需要使用返回泛型的,那我们就使用out,如果我们需要将泛型放入方法中使用的,我们就是用in

    interface Production<out T> {
          fun produce(): T
      }
      
    interface Consumer<in T> {
          fun consume(item: T)
      }
    

    如果过程中我们既要使用返回泛型,又要使用泛型使用,我们就直接使用使用T即可,即既没有in也没有out,这应该是大部分时间会出现的场景。

    我们继续结合业务场景,进行针对协变逆变和in``out做更深度的解读。

    我们照着上面两个接口的条件,先写下如下代码:

    class Banana
    
    class BananaFactory : Production<Banana> {
          override fun produce(): Banana {
              return Banana()
          }
      }
      
    var factory : Production<Banana> = BananaFactory()
    

    香蕉工厂,产香蕉。直接单单这么看,不会有很深刻的印象。我们把banana换成有依赖关系的三种产品。

    //食物
    open class Food
    //快餐是食物的一种
    open class FastFood : Food()
    //汉堡是快餐的一种
    class Burger : FastFood()
    

    针对这三种产品,有三个相应的工厂

     class FoodFactory : Production<Food> {
          override fun produce(): Food {
              return Food()
          }
      }
      
      class FastFoodFactory : Production<FastFood> {
          override fun produce(): FastFood {
              return FastFood()
          }
      }
      
      class BurgerFactory : Production<Burger> {
          override fun produce(): Burger {
              return Burger()
          }
      }
    

    然后我们针对三种工厂开始进行赋值,首先是显而易见肯定没问题的赋值。

     val production1 : Production<Food> = FoodFactory()
      val production2 : Production<FastFood> = FastFoodFactory()
      val production3 : Production<Burger> = BurgerFactory() 
    

    然后我们尝试这种Food去接所有的品类工厂,发现并没有报错。

     val production1 : Production<Food> = FoodFactory()
      val production2 : Production<Food> = FastFoodFactory()
      val production3 : Production<Food> = BurgerFactory()
    

    我们再尝试着用Burger去接,发现除了最后一个可以接,上面的统统报错。

     val production1 : Production<Burger> = FoodFactory()  //报错
      val production2 : Production<Burger> = FastFoodFactory()   //报错
      val production3 : Production<Burger> = BurgerFactory()
    

    FoodFactory、FastFoodFactory、BurgerFactory三个工厂生产的都是Food,但是只有BurgerFactory生产的肯定是BurgerFastFoodFactory生产的可能还有别的快餐,FoodFactory生产的可能还有别的食物。

我们回头再来看Consumer的接口的三个实现。

//所有事物都吃
class FoodConsumer : Consumer<Food> {
      override fun consume(item: Food) {
      }
  }
  
  //所有快餐都吃
  class FastFoodConsumer : Consumer<FastFood> {
      override fun consume(item: FastFood) {
      }
  }
  
  //所有汉堡都吃
  class BurgerConsumer : Consumer<Burger> {
      override fun consume(item: Burger) {
      }
  }

首先,各自对应生成自己的Consumer<?>肯定是没有问题的,这里我们代码就不写了,我们直接看两个封顶首先是Food,发现一上来就编译出错了

 val consumer1 : Consumer<Food> = FoodConsumer()
  val consumer2 : Consumer<Food> = FastFoodConsumer()   //报错
  val consumer3 : Consumer<Food> = BurgerConsumer()   //报错

如何解释,所有食物都吃的人,肯定是吃Food的。所以第一个没问题。只吃快餐的人FastFoodConsumer,并不一定吃所有的Food,只吃汉堡的人BurgerConsumer就更不用说了。那么可想而知,反过来应该是都通过了。

 val consumer1 : Consumer<Burger> = FoodConsumer()
  val consumer2 : Consumer<Burger> = FastFoodConsumer()
  val consumer3 : Consumer<Burger> = BurgerConsumer()

OK,至此,我们故事讲完了,开始强化理解。

  • 我们在没加inout之前,直接使用泛型T,效果和在java中的是一样一样的。

    即针对所有的跟T有继承关系的父类或者子类,都无法直接突破,即不变

  • out表示子类泛型对象都可以赋值给父类泛型对象,大的包含小的,小的要到大的里面去是outout只返回T,不消费T

  • in表示父类泛型对象都可以赋值给子类泛型对象,大的包含小的,大的要进到小的里面去是inin只消费T,不返回T

  • inout是kotlin所独有的,在java中,没有直接申明继承关系的情况下是无法直接进行泛型对象互相赋值的。

  • Array<? extends Food>直接用Array<out Food>,即所有Food的子类都可以添加到这个数组中去,同时所有的Array<Food子类>都可以赋值给定义为Array<out Food>的对象

  • 星号投射

    kotlin中,我们用*代替所有的类型,相当于Any?。不用过多的去跟inoutNothing等等的做挂钩,只需要知道*基本就相当于java中的object,即可以代表任何的类型。

    举个例子:

    class A<T>(val t: T, val t2 : T, val t3 : T)
    val a1: A<*> = A(12, "String", Apple("苹果"))
    

    在kotlin中,我们一般其实不会想要去申明对象的类型,而交给系统自己去判断,目前的用途还没有特别清楚,待之后补充。


Kotlin学习笔记之 1 基础语法

Kotlin学习笔记之 2 基本数据类型

Kotlin学习笔记之 3 条件控制

Kotlin学习笔记之 4 循环控制

Kotlin学习笔记之 5 类和对象

Kotlin学习笔记之 6 继承

Kotlin学习笔记之 7 接口

Kotlin学习笔记之 8 扩展

Kotlin学习笔记之 9 数据类与密封类

Kotlin学习笔记之 10 泛型

Kotlin学习笔记之 11 枚举类

Kotlin学习笔记之 12 对象表达式和对象声明

Kotlin学习笔记之 13 基础操作符run、with、let、also、apply

Kotlin学习笔记之 14 包与导入

Kotlin学习笔记之 15 伴生对象

Kotlin学习笔记之 16 委托

Kotlin学习笔记之 17 可观察属性

Kotlin学习笔记之 18 函数

Kotlin学习笔记之 19 高阶函数与 lambda 表达式

Kotlin学习笔记之 20 内联函数

Kotlin学习笔记之 21 解构声明

Kotlin学习笔记之 22 集合

Kotlin学习笔记之 23 相等判断

Kotlin学习笔记之 24 操作符重载

Kotlin学习笔记之 25 异常捕捉

Kotlin学习笔记之 26 反射

Kotlin学习笔记之 27 类型别名

Kotlin学习笔记之 28 协程基础

Kotlin学习笔记之 29 上下文与调度器

Kotlin学习笔记之 30 协程取消与超时

Kotlin学习笔记之 31 协程挂起函数的组合

Kotlin学习笔记之 32 协程异常处理

Kotlin学习笔记之 33 协程 & Retrofit

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 156,265评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,274评论 1 288
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,087评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,479评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,782评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,218评论 1 207
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,594评论 2 309
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,316评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 33,955评论 1 237
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,274评论 2 240
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,803评论 1 255
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,177评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,732评论 3 229
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,953评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,687评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,263评论 2 267
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,189评论 2 258

推荐阅读更多精彩内容