美文网首页
Kotlin基础语法<二>

Kotlin基础语法<二>

作者: 安仔夏天勤奋 | 来源:发表于2020-02-09 17:06 被阅读0次

    学习需要一步步进行,前面已学习过了部分kotlin基础语法<Kotlin基础语法一>
    。基础语法的学习与总结会让我对知识了了解更加深入,也算是对自己的激励吧。

    Kotlin-数组

    1. 数组在kotlin中的使用Array<T>类来表示
    2. 基本类型数组:ByteArray,LongArray,xxxArray等,这些类是基本类型数组,但是跟Array类没有继承关系
    3. 数组的创建
      • 使用arrayOf()函数(java的静态初始化)
      • 使用arrayOfNulls()函数(java的动态初始化)
      • 使用emptyArray()函数
      • 使用Array(size:Int,init:(int)->T)构造器
    fun main(){
        val arr = intArrayOf(1,2,20,36,45,8)
        for((index,value) in arr:withIndex()){
            println("索引为${index}的元素是:${value}")
        }
    }
    

    Kotlin的集合

    java中的类型 Kotlin中的只读类型 Kotlin中的可变类型
    Iterator<T> Iterator<T> MutableIterator<T>
    Iterable<T> Iterable<T> MutableIterable<T>
    Collection<T> Collection<T> MutableCollection<T>
    Set<T> Set<t> MutableSet<t>
    List<T> List<T> MutableList<T>
    ListIterator<T> ListIterator<T> MutableListIterator<T>
    Map<K,V> Map<K,V> MutableMap<K,V>
    Map.Enrty<K,V> Map.Entry<K,V> MutableMap.MutableEntry<K,V>
    1. kotlin集合可分为可变和不可变集合
    2. 声明并初始化List的集合,使用listOf(..)函数
    3. 声明并初始化MutableList的集合,使用mutableListOf(..)函数/listOfNotNull()/arrayListOf()
    4. 声明并初始化Set集合,使用setOf(..)函数
    5. 声明并初始化MutableSet的集合,使用mutableSetOf(..)函数/hashSetOf()/linkedSetOf()/sortedSetOf()
    6. 不可变的Map类型的初始化,使用mapOf()函数
    7. 可变的Map类型集合的初始化,使用mutableMapOf()函数/hashMapOf()/linkedMapOf()/sortedMapOf()

    for循环使用

    fun main() {
        val items = listOf("java", "kotlin", "android")
        for (item  in items) {
            println(item)
        }
    }
    
    fun main() {
        val items = listOf("java", "kotlin", "android")
        for (index  in items.indices) {
            println("item at $index is ${items[index]}")
        }
    }
    

    Kotlin的泛型

    先了解一下java的泛型

    上界通配符 < ? extends X>。用 extends 关键字声明,表示参数化的类型可能是所指定的类型,或者是此类型的子类。

    上界 ->取出来的类型不会丢失,存放会丢失类型。(?代表容器里的元素类型为X基类类型,X是所有元素的基类,存(即set)进去就无法确定是那个具体的类型了,取出来就没有问题,因为X是表示所有基类)

    下界通配符 < ? super E>。用 super 进行声明,表示参数化的类型可能是所指定的类型,或者是此类型的父类型,直至 Object。

    下界 ->取出来的类型会丢失,存放不会丢失类型。(?代表容器里的元素类型为E基类类型,存(即set)进去的都是E的超类,取出来就不知道是那个具体的类型,这样就没法统一基类类型了,全部都为Object)

    上界、下界遵循PECS (Producter Extends Consumer Super)

    Producter 产生者 -> 取
    Consumer 消费者 -> 存

    详细的java泛型

    Kotlin的泛型 jvm泛型从java中继续过来的

    out(相当于java的上界)取出来是没有问题,存进去就有问题。 out在Kotlin中叫协变。

    in(相当于java的下界)存进去没有问题,取出来就是问题。 in在Kotlin中叫逆变。

    有一个copy数组的函数,看如下代码:

    fun copy(destArr:Array<Double>,srcArr:Array<Double>){
    }
    
    fun main(){
        var destDouble = arrayOf<Double>()
        var srcDouble = arrayOf<Double>(1.1,2.2,3.3)
        copy(destDouble,srcDouble)
    }
    

    如果有多种类型的数组要copy,那么copy函数就得复写多个。复写多个copy函数,这样写代码臃肿,也不优雅。看下面代码如何优雅实现根据不同类型实现copy函数。

    //泛型函数   泛型
    fun <T> copy(destArr:Array<T>,srcArr:Array<T>){
    }
    
    fun main(){
        var destDouble = arrayOf<Double>()
        var srcDouble = arrayOf<Double>(1.1,2.2,3.3)
        copy(destDouble,srcDouble)
    
        var destInt = arrayOf<Int>()
        var srcInt = arrayOf(1,2,3)
        copy(destInt,srcInt)
    }
    

    不错上述的copy函数就是使用了泛型,也是一个简单的使用。copy这个泛型函数,还是存在问题的,Int/Double的父类为Number。如果destArrc参数传入一个arrayOf<Number>数组,那么copy函数就报错了。这个问题也很容易解决,把srcArr存放到destArr数组中,那么把destArr的泛型改为逆变就可以了(destArr:Array<in T>)。如下代码:

    fun <T> copy(dest:Array<in T>,src:Array<T>){
        //把srcArr的内容拿出来,存放到destArr数组中
        srcArr.forEachIndexed{ index,value->destArr[index] = srcArr[index]}
    }
    fun main(){
        var srcInt = arrayOf(1,2,3)
        var destNum = arrayOf<Number>(3)
        copy(destNum,srcInt)
    }
    

    上述的dest:Array<in T>,in T ( 相当于java中的 ?super T)是一个逆变。逆变之后,可以把Array<Number>理解为Array<Int>/Array<Double>的子类。上面的copy泛型函数写法还写成另一种,代码如下:

    fun <T> copy(dest:Array<T>,src:Array<out T>){
        //把srcArr的内容拿出来,存放到destArr数组中
        srcArr.forEachIndexed{ index,value->destArr[index] = srcArr[index]}
    }
    
    fun main(){
        var srcInt = arrayOf(1,2,3)
        var destNum = arrayOf<Number>(3)
        copy(destNum,srcInt)
    }
    

    copy(dest:Array<T>,src:Array<out T>)是协变的写法。src:Array<out T>取出来的类型一定是T类型,所以存放到destArr<T>是没有问题的。

    特别例子:在java不允许泛型使用方式,而在kotlin中可以使用

    java中不允许泛型使用的方式,代码如下:

    public class A{
        interface Callback<T>{//声明处形变 
            T getT();
        }
        interface Call<? extends K>{//声明处形变 是不允许这样使用的  无法声明处形变的 
        }
        void test(Callback<String> callback){
            Callback<String> cb = callback;//java 是不允许这样使用的
        }
    }
    

    kotlin中可以使用java中不允许泛型使用的方式,代码如下:

    interface Callback<out T>{
        fun getT():T
    }
    
    fun test(src:Callback<String>){
        val dest:Callback<Any?> = src
    }
    

    从上述两处代码得出:kotlin对java的泛型的增强,多了一个声明处形变。

    Kotlin的lambda

    { }就是表示一个lambda表达式

    fun main(){
        var sum ={x:Int,y:Int -> x+y}
        //lambda表达式调用
        com.lu.kotlindemo.testFun.sum(1,3) 
    }
    
    fun main(){
        var temp:(Int,Int)->Int  //相当于 var temp:Any
        //这是一个lambda { }
        temp ={x:Int,y:Int -> x+y}
        val sum = temp.invoke(2,2)
        println("相加:${sum}")
    
        temp ={x:Int,y:Int -> x*y}
        val c = temp.invoke(3,4)
        println("相乘:${c}")
    
        //这也是一个lambda
        var tmp :(Int)->Unit 
        tmp ={ println("$it")}//这个it代表只有一个参数(默认只有一个参数)
    }
    
    

    var temp:(Int,Int)->Int 。声明处用括号括起来,也是就是定义一个类型 ,传入两个Int类型参数并返回一个Int类型值。相当于 var temp:Any。

    tmp ={ println("$it")},注意:这个it代表只有一个参数(默认只有一个参数)。如果有返回值,最后一句就是代表返回值,不能要{}里面return。

    错误代码:

    tmp ={
        println("$it")
        return //不能够这样写return
    }
    
    

    正确代码:

    tmp ={
        println("$it")
        it+3 //最后一句 代表返回值
    }
    
    

    相关文章

      网友评论

          本文标题:Kotlin基础语法<二>

          本文链接:https://www.haomeiwen.com/subject/wxzrzctx.html