第2章 数据类型

        如果把写程序比喻成盖房子,那么各种变量相当于各种建筑材料,建材包括砖头、水泥、沙子等,程序变量也分为不同的数据类型,例如整型数、浮点数、数组、字符串以及更高级的各种容器等。建筑内容有诸如砌砖头、搅拌泥沙等处理操作,数据类型也拥有自己的常见操作,如转换、修改、查询等。下面就依次介绍如何使用Kotlin的各种数据类型。

2.1 基本数据类型

        每个编程语言都离不开基本的数据类型,包括整型、浮点型、布尔型等,当然Kotlin也不例外。虽然基本数据类型的概念是老生常谈,但是Kotlin声明基本变量究竟有哪些特别之处呢?本节从基本变量类型开始,逐步探讨这些数据类型的常见用法。

2.1.1 基本类型的变量声明

        Kotlin的基本数据类型跟其他高级语言的分类一样,包括整型、长整型、浮点型、双精度、布尔型、字符型、字符串这几种常见类型,具体的类型名称说明见表2-1。

        看起来很熟悉是不是,Kotlin原来这么简单。可是如果你马上敲出变量声明的代码,便会发现编译有问题。比如声明一个最简单的整型变量,按Java的写法是下面这样:

int i=0;

        倘若按照Java的规则来书写Kotlin代码,就是下面这行代码:

Int i=0;

        然而Android Studio立即提示编译不通过,刚开始学Kotlin便掉到坑里,看来要认真对待Kotlin,不能这么轻易让它坑蒙拐骗了。正确的Kotlin声明变量的代码是下面这样的:

var i:Int = 0

        前面的var表示后面是一个变量声明语句,接着是“变量名:变量类型”的格式声明,而不是常见的“变量类型 变量名”这种格式。至于后面的分号,则看该代码行后面是否还有其他语句,如果变量声明完毕直接回车换行,那么后面无须带分号;如果没有回车换行,而是添加其他语句,那么变量声明语句要带上分号。

2.1.2 简单变量之间的转换

        Kotlin变量的另一个重要特点是类型转换,在Java开发中,如int、long、float、double类型的变量可以直接在变量名前面加上诸如(int)、(long)、(float)、(double)这种表达式进行强制类型转换;对于int(整型)和char(字符型)这两种类型,甚至都无须转换类型,直接互相赋值即可。但在Kotlin中,不允许通过Java的前缀表达式来强制转换类型,只能调用类型转换函数输出其他类型的变量,表2-2是常见的几种类型转换函数的说明。

        接下来通过实际代码观察一下类型转换的过程,测试用到的类型转换的Kotlin代码片段如下所示:

val origin:Float = 65.0f
tv_origin.text = origin.toString()
var int:Int
btn_int.setOnClickListener { int=origin.toInt(); tv_convert.text=int.toString() }
var long:Long
btn_long.setOnClickListener { long=origin.toLong(); tv_convert.text=long.toString() }
var float:Float
btn_float.setOnClickListener { float=origin.toDouble().toFloat(); tv_convert.text=float.toString() }
var double:Double
btn_double.setOnClickListener { double=origin.toDouble(); tv_convert.text=double.toString() }
var boolean:Boolean
btn_boolean.setOnClickListener { boolean=origin.isNaN(); tv_convert.text=boolean.toString() }
var char:Char
btn_char.setOnClickListener { char=origin.toChar(); tv_convert.text=char.toString() }

        各种类型转换的操作结果如图2-1~图2-3所示,其中图2-1展示转换为整型的界面效果,图2-2展示转换为双精度的界面效果,图2-3展示转换为字符型的界面效果。

        注意到上述类型转换代码的第一行变量声明语句以val开头,而其余的变量声明语句均以var开头,这是为什么呢?其实val和var的区别在于,前者修饰过的变量只能在第一次声明时赋值,后续不能再赋值;而后者修饰过的变量在任何时候都允许赋值。方便记忆的话,可以把val看作是Java里的final关键字;至于var,Java里面没有对应的关键字,就当它是例行公事好了。

2.2 数组

        2.1节介绍了基本数据类型在Kotlin中的用法,不过这只针对单个变量,如果要求把一组相同类型的变量排列起来,形成一个变量数组,那又该如何声明和操作呢?本节就来谈谈Kotlin对数组的常见用法。

2.2.1 数组变量的声明

        在Java中声明数组跟在C语言中声明是一样的,以整型数组为例,声明数组并加以初始化的语句如下所示:

int[] int_array = new int[] {1, 2, 3};

        其他基本类型的数组声明与之类似,只要把int替换为long、float、double、boolean、char其中之一即可。但在Kotlin中,声明并初始化一个整型数组的语句是下面这样的:

var int_array:IntArray = intArrayOf(1, 2, 3)

        两相对比,对于整型数组的声明,Kotlin与Java之间有以下区别:

        (1)Kotlin另外提供了新的整型数组类型,即IntArray。

        (2)分配一个常量数组,Kotlin调用的是intArrayOf方法,并不使用new关键字。

        推而广之,其他基本类型的数组也各有自己的数组类型,以及对应分配常量数组的初始化方法,详细的对应关系说明见表2-3。

        下面是这些基本类型数组的初始化代码例子:

var long_array:LongArray = longArrayOf(1, 2, 3)
var float_array:FloatArray = floatArrayOf(1.0f, 2.0f, 3.0f)
var double_array:DoubleArray = doubleArrayOf(1.0, 2.0, 3.0)
var boolean_array:BooleanArray = booleanArrayOf(true, false, true)
var char_array:CharArray = charArrayOf('a', 'b', 'c')

        不知读者有没有注意到,上面的Kotlin数组类型不包括字符串数组,而Java是允许使用字符串数组的,声明字符串数组的Java代码示例如下:

String[] string_array = new String[] {"How", "Are", "You"};

        但在Kotlin这里,并不存在名为StringArray的数组类型,因为String是一种特殊的基本数据类型。要想在Kotlin中声明字符串数组,得使用Array<String>类型,也就是把“String”用尖括号包起来。同时,分配字符串数组的方法也相应变成了arrayOf,下面是声明字符串数组的Kotlin代码:

var string_array:Array<String> = arrayOf("How", "Are", "You")

        这种字符串数组的声明方式是不是很熟悉?看起来就跟Java里面的ArrayList用法差不多,都是在尖括号中间加入数据结构的类型。同理,其他类型的数组变量也能通过“Array<数据类型>”的方式来声明,像前面介绍的整型数组,其实可以使用类型Array<Int>,以此类推,改造之后的各类型数组变量的声明代码如下所示:

var int_array:Array<Int> = arrayOf(1, 2, 3)
var long_array:Array<Long> = arrayOf(1, 2, 3)
var float_array:Array<Float> = arrayOf(1.0f, 2.0f, 3.0f)
var double_array:Array<Double> = arrayOf(1.0, 2.0, 3.0)
var boolean_array:Array<Boolean> = arrayOf(true, false, true)
var char_array:Array<Char> = arrayOf('a', 'b', 'c')

2.2.2 数组元素的操作

        现在声明数组和对数组初始化的代码都有了,接下来还需要对数组做进一步的处理,常见的处理包括获取数组长度、获取指定位置的数组元素等,这些操作在Kotlin与Java之间的区别包括:

        (1)对于如何获取数组长度,Java使用.length,而Kotlin使用.size。

        (2)对于如何获取指定位置的数组元素,Java通过方括号加下标来获取,比如“int_array[0]”指的是得到该数组的第一个元素;Kotlin也能通过方括号加下标来获取指定元素,不过Kotlin还拥有get和set两个方法,通过get方法获取元素值,通过set方法修改元素值,看起来就像在操作ArrayList队列。

下面是Kotlin操作字符串数组的示例代码:
//声明字符串数组
var string_array:Array<String> = arrayOf("How", "Are", "You")
btn_string.setOnClickListener {
    var str:String = ""
    var i:Int = 0
    while (i<string_array.size) {
        str = str + string_array[i] + ", "
        //数组元素可以通过下标访问,也可通过get方法访问
        //str = str + string_array.get(i) + ", "
        i++
    }
    tv_item_list.text = str
}

        上述代码的演示效果如图2-4所示,可以看到字符串数组内部的各元素都被逗号分隔开了。

2.3 字符串

        2.2节介绍了数组的声明和操作,其中包括字符串数组的用法。注意到Kotlin的字符串类型名称跟Java一样都叫String,那么字符串在Kotlin和Java中的用法有哪些差异呢?这便是本节所要阐述的内容了。

2.3.1 字符串与基本类型的转换

        首先要说明的是字符串类型与基本变量类型之间的转换方式,在前面的“2.1.2 简单变量之间的转换”中,提到基本数据类型的变量可以通过toString方法转换为字符串类型。反过来,字符串类型又该如何转换为基本变量类型呢?表2-4展示使用Kotlin和Java编码将字符串转换为基本数据类型的对照方式说明。

        就表2-4的转换情况来看,Java的实现方式比较烦琐,既需要其他类型的类名,又需要该类型的转换方法。而在Kotlin这边,转换类型相对简单,并且与基本数据类型之间的转换形式保持一致,即都是采取“to***()”的形式。显而易见,Kotlin对字符串的类型转换方式更友好,也更方便记忆。

2.3.2 字符串的常用方法

        当然,转换类型只是字符串的基本用法,还有更多处理字符串的其他用法,比如查找子串、替换子串、截取指定位置的子串、按特定字符分隔子串等,在这方面Kotlin基本兼容Java的相关方法。对于查找子串的操作,二者都调用indexOf方法;对于截取指定位置子串的操作,二者都调用substring方法;对于替换子串的操作,二者都调用replace方法;对于按特定字符分隔子串的操作,二者都调用split方法。

        下面是Kotlin使用indexOf和substring方法进行子串查找和截取字符串的代码例子:

//截取小数点之前的字符串,即取整操作
val origin:String = tv_origin.text.toString()
var origin_trim:String = origin
if (origin_trim.indexOf('.') > 0) {
    origin_trim = origin_trim.substring(0, origin_trim.indexOf('.'))
}

        在这些字符串处理方法里面,唯一区别是split方法的返回值,在Java中,split方法返回的是String数组,即String[];但在Kotlin中,split方法返回的是String队列,即List<String>。下面是Kotlin使用split方法的示例代码:

//根据点号将源串分割为字符串队列,并将分割结果显示在界面上
btn_split.setOnClickListener {
    var strList:List<String> = origin.split(".")
    var strResult:String = ""
    for (item in strList) {
        strResult = strResult + item + ", "
    }
    tv_convert.text = strResult
}

        分割字符串的界面效果如图2-5所示,可以看到源字符串里面的点号都被替换为逗号,字符串末尾也多了一个逗号。

        若想获取字符串某个位置的字符,这个看似简单的需求,采取Java实现时却有点烦琐,因为只能调用substring方法去截取指定位置的字符串,具体的Java代码如下所示:

String result = origin.substring(number, number+1);
tv_convert.setText(result);

        通过Kotlin实现上述需求就简单多了,因为Kotlin允许直接通过下标访问字符串指定位置的字符,下面是访问字符串指定位置的Kotlin代码例子:

tv_convert.text = origin[number].toString()

        同时,Kotlin也支持字符串变量通过get方法获取指定位置上的字符,代码如下:

tv_convert.text = origin.get(number).toString()

        如此一来,Kotlin的字符串定位代码不但更加精炼,而且可读性也增强了。

2.3.3 字符串模板及其拼接

        Kotlin对字符串带来的便利并不限于此,举个例子,若Java把几个变量拼接成字符串,则要么用加号强行拼接,要么用String.format函数进行格式化。可是前者的拼接加号时常会跟数值相加的加号混淆;而后者的格式化还得开发者死记硬背,如%d、%f、%s、%c、%b等格式转换符,实在令人头痛。对于字符串格式化这个痛点,Kotlin恰如其分地进行了优化,何必引入这些麻烦的格式转换符呢?直接在字符串中加入“$变量名”即可表示此处引用该变量的值,岂不妙哉!

        心动不如行动,赶紧动起手来,看看Kotlin如何格式化字符串,先来看一个示例代码:

btn_format.setOnClickListener { tv_convert.text = "字符串值为 $origin" }

        这里要注意,符号$后面跟变量名,系统会自动匹配最长的变量名。比如下面这行代码,打印出来的是变量origin_trim的值,而不是origin的值:

btn_format.setOnClickListener { tv_convert.text = "字符串值为 $origin_trim" }

        另外,有可能变量会先进行运算,再把运算结果拼接到字符串中。此时,需要用大括号把运算表达式给括起来,具体代码如下所示:

btn_length.setOnClickListener { tv_convert.text = "字符串长度为 ${origin.length}" }

        在上述的Kotlin格式化代码中,美元符号$属于特殊字符,因此不能直接打印它,必须经过转义才可以打印。转义的办法是使用“${'***'}”表达式,该表达式外层的“${''}”为转义声明,内层的“***”为需要原样输出的字符串,所以通过表达式“${'$'}”即可打印一个美元符号,示例代码如下:

btn_dollar.setOnClickListener { tv_convert.text = "美元金额为 ${'$'}$origin" }

        如果只是对单个美元符号做转义,也可直接在符号$前面加个反斜杆,即变成“\$”,修改后的代码如下所示:

btn_dollar.setOnClickListener { tv_convert.text = "美元金额为 \$$origin" }

        然而一个反斜杆仅仅对一个字符进行转义,倘若要对一个字符串做转义,也就是把某个字符串的所有字符原样输出,那么只能采用形如“${'***'}”的表达式,该表达式利用单引号把待转义的字符串包起来,好处是能够保留该字符串内部的所有特殊字符。

2.4 容器

        Kotlin号称全面兼容Java,于是Java的容器类仍可在Kotlin中正常使用,包括大家熟悉的队列ArrayList、映射HashMap等。不过Kotlin作为一门全新的语言,肯定要有自己的容器类,不然哪天Java跟Kotlin划清界限,那麻烦就大了。本节就对Kotlin的几种容器类进行详细的说明。

2.4.1 容器的基本操作

        与Java类似,Kotlin也拥有三类基本的容器,分别是集合Set、队列List、映射Map,然后每类容器又分作只读与可变两种类型,这是为了判断该容器能否进行增、删、改等变更操作。Kotlin对变量的修改操作很慎重,每个变量在定义的时候就必须指定能否修改,比如添加val修饰表示该变量不可修改,添加var修饰表示该变量允许修改。至于容器则默认为只读容器,如果需要允许修改该容器变量,就需要加上Mutable前缀形成新的容器,比如MutableSet表示可变集合,MutableList表示可变队列,MutableMap表示可变映射,只有可变的容器才能够对其内部元素进行增、删、改操作。

        既然集合Set、队列List、映射Map三者都属于容器,那么它们必定拥有相同的容器方法,这些公共方法具体说明如下。

        isEmpty:判断该容器是否为空。
        isNotEmpty:判断该容器是否非空。
        clear:清空该容器。
        contains:判断该容器是否包含指定元素。
        iterator:获取该容器的迭代器。
        count:获取该容器包含的元素个数,也可通过size属性获得元素数量。

        另外,Kotlin允许在声明容器变量时就进行初始赋值,如同对数组变量进行初始化那样。而Java的容器类是无法同时声明并初始化的,由此可见Kotlin的这点特性给开发者带来很大便利。下面是一个初始化List队列的Kotlin代码例子:

val satellites:List<String> = listOf("水星", "金星", "地球", "火星", "木星", "土星")

        当然,不同容器的初始化方法有所区别,各种容器与其初始化方法的对应关系见表2-5。

        以上介绍了Kotlin容器的基本用法,更具体的增、删、改、查等操作则有所不同,接下来分别说明这三类6种容器的详细使用。

2.4.2 集合Set/MutableSet

        集合是一种最简单的容器,它具有以下特性:

        (1)容器内部的元素不按顺序排列,因此无法按照下标进行访问。

        (2)容器内部的元素存在唯一性,通过哈希值校验是否存在相同的元素,若存在,则将其覆盖。

        因为Set是只读集合,初始化赋值后便不可更改,所以元素变更的方法只适用于可变集合MutableSet,但MutableSet的变更操作尚有以下限制:

        (1)MutableSet的add方法仅仅往集合中添加元素,由于集合是无序的,因此不知道添加的具体位置。

        (2)MutableSet没有修改元素值的方法,一个元素一旦被添加,就不可被修改。

        (3)MutableSet的remove方法用于删除指定元素,但无法删除某个位置的元素,这是因为集合内的元素不是按顺序排列的。

        对于集合的遍历操作,Kotlin提供了好几种方式,有熟悉的for-in循环、迭代器遍历,还有新面孔forEach遍历,这三种集合遍历的用法说明如下。

1. for-in循环

        与Java类似,通过for语句加上in条件即可轻轻松松依次取出集合中的所有元素。下面是运用for-in循环的代码例子:

val goodsMutSet:Set<String> = setOf("iPhone8", "Mate10", "小米6", "OPPO R11", "vivo X9S", "魅族Pro6S")
btn_set_for.setOnClickListener {
    var desc = ""
    //使用for-in语句循环取出集合中的每条记录
    for (item in goodsMutSet) {
        desc = "${desc}名称:${item}\n"
    }
    tv_set_result.text = "手机畅销榜包含以下${goodsMutSet.size}款手机:\n$desc"
}

        上述代码对应的界面效果如图2-6所示,可见初始化时输入的6部手机都通过遍历操作打印了出来。

2. 迭代器遍历

        迭代器与指针的概念有点接近,它自身并非具体的元素,而是指向元素的存放地址,所以迭代器遍历其实是遍历所有元素的地址。迭代器通过hasNext方法判断是否还存在下一个节点,如果不存在下一节点,就表示已经遍历完毕,它通过next方法获得下一个节点的元素,同时迭代器自身改为指向该元素的地址。下面是运用迭代器遍历的代码例子:

btn_set_iterator.setOnClickListener {
    var desc = ""
    val iterator = goodsMutSet.iterator()
    //如果迭代器还存在下一个节点,就继续取出下一个节点的记录
    while (iterator.hasNext()) {
        val item = iterator.next()
        desc = "${desc}名称:${item}\n"
    }
    tv_set_result.text = "手机畅销榜包含以下${goodsMutSet.size}款手机:\n$desc"
}

3. forEach遍历

        无论是for-in循环还是迭代器遍历,其实都脱胎于Java已有的容器遍历操作,代码书写上不够精炼。为了将代码精简到极致,Kotlin给容器创造了forEach方法,明确指定该方法就是要依次遍历容器内部的元素。forEach方法在编码时采用匿名函数的形式,内部使用it代表每个元素,下面是运用forEach遍历的代码例子:

btn_set_foreach.setOnClickListener {
    var desc = ""
    //forEach内部使用it指代每条记录
    goodsMutSet.forEach { desc = "${desc}名称:${it}\n" }
    tv_set_result.text = "手机畅销榜包含以下${goodsMutSet.size}款手机:\n$desc"
}

        结合以上有关Set/MutableSet的用法说明,可以发现集合在实战中存在诸多不足,主要包括以下几点:

        (1)集合不允许修改内部元素的值。

        (2)集合无法删除指定位置的元素。

        (3)不能通过下标获取指定位置的元素。

        鉴于集合的以上缺点难以克服,故而实际开发基本用不到集合,大多数场合用的是它的两个兄弟——队列和映射。

2.4.3 队列List/MutableList

        队列是一种元素之间按照顺序排列的容器,它与集合的最大区别在于多了次序管理。不要小看这个有序性,正因为队列建立了秩序规则,所以它比集合多提供了如下功能(注意,凡是涉及增、删、改的,都必须由MutableList来完成):

        (1)队列能够通过get方法获取指定位置的元素,也可以直接通过下标获得该位置的元素。

        (2)MutableList的add方法每次都是把元素添加到队列末尾,也可指定添加的位置。

        (3)MutableList的set方法允许替换或者修改指定位置的元素。

        (4)MutableList的removeAt方法允许删除指定位置的元素。

        (5)队列除了拥有跟集合一样的三种遍历方式(for-in循环、迭代器遍历、forEach遍历)外,还多了一种按元素下标循环遍历的方式,具体的下标遍历代码例子如下:

val goodsMutList:List<String> = listOf("iPhone8", "Mate10", "小米6", "OPPO R11", "vivo X9S", "魅族Pro6S")
btn_for_index.setOnClickListener {
    var desc = ""
    //indices是队列的下标数组。如果队列大小为10,下标数组的取值就为0~9
    for (i in goodsMutList.indices) {
        val item = goodsMutList[i]
        desc = "${desc}名称:${item}\n"
    }
    tv_list_result.text = "手机畅销榜包含以下${goodsMutList.size}款手机:\n$desc"
}

        上面按下标遍历队列的代码对应的运行界面如图2-7所示,可见队列中保存的6部手机都通过遍历显示了出来。

        (6)MutableList提供了sort系列方法用于给队列中的元素重新排序,其中sortBy方法表示按照指定条件升序排列,sortByDescending方法表示按照指定条件降序排列。下面是一个给队列排序的代码例子(含升序和降序):

var sortAsc = true
btn_sort_by.setOnClickListener {
    if (sortAsc) {
        //sortBy表示升序排列,后面跟的是排序条件
        goodsMutList.sortBy { it.length }
    } else {
        //sortByDescending表示降序排列,后面跟的是排序条件
        goodsMutList.sortByDescending { it.length }
    }
    var desc = ""
    for (item in goodsMutList) {
        desc = "${desc}名称:${item}\n"
    }
    tv_list_result.text = "手机畅销榜已按照${if (sortAsc) "升序" else "降序"}重新排列:\n$desc"
    sortAsc = !sortAsc
}

        队列进行排序操作后的演示效果如图2-8和图2-9所示,其中图2-8展示按升序排列后的手机信息界面,图2-9展示按降序排列后的手机信息界面。

2.4.4 映射Map/MutableMap

        映射内部保存的是一组键值对(Key-Value),也就是说,每个元素都由两部分构成,第一部分是元素的键,相当于元素的名字;第二部分是元素的值,存放着元素的详细信息。元素的键与值是一一对应的关系,相同键名指向的键值是唯一的,所以映射中每个元素的键名各不相同,这个特性使得映射的变更操作与队列存在以下不同之处(注意,增、删操作必须由MutableMap来完成):

        (1)映射的containsKey方法判断是否存在指定键名的元素,containsValue方法判断是否存在指定键值的元素。

        (2)MutableMap的put方法不单单是添加元素,而是智能的数据存储。每次调用put方法时,映射会先根据键名寻找同名元素,如果找不到就添加新元素,如果找得到就用新元素替换旧元素。

        (3)MutableMap的remove方法是通过键名来删除元素的。

        (4)调用mapOf和mutableMapOf方法初始化映射时,有两种方式可以表达单个键值对元素,其一是采取“键名 to 键值”的形式,其二是采取Pair配对方式,形如“Pair(键名, 键值)”。下面是这两种初始化方式的代码例子:

//to方式初始化映射
var goodsMap: Map<String, String> = mapOf("苹果" to "iPhone8", "华为" to "Mate10", "小米" to "小米6", "欧珀" to "OPPO R11", "步步高" to "vivo X9S", "魅族" to "魅族Pro6S")
//Pair方式初始化映射
var goodsMutMap: MutableMap<String, String> = mutableMapOf(Pair("苹果", "iPhone8"), Pair("华为", "Mate10"), Pair("小米", "小米6"), Pair("欧珀", "OPPO R11"), Pair("步步高", "vivo X9S"), Pair("魅族", "魅族Pro6S"))

        映射的遍历与集合类似,也有for-in循环、迭代器遍历、forEach遍历三种遍历手段。但是由于映射的元素是一个键值对,因此它的遍历方式与集合稍有不同,详述如下:

1. for-in循环

        for-in语句取出来的是映射的元素键值对,若要获取该元素的键名,还需访问元素的key属性;若要获取该元素的键值,还需访问元素的value属性。下面是在映射中运用for-in循环的代码例子:

btn_map_for.setOnClickListener {
    var desc = ""
    //使用for-in语句循环取出映射中的每条记录
    for (item in goodsMutMap) {
        //item.key表示该配对的键,即厂家名称;item.value表示该配对的值,即手机名称
        desc = "${desc}厂家:${item.key},名称:${item.value}\n"
    }
    tv_map_result.text = "手机畅销榜包含以下${goodsMutMap.size}款手机:\n$desc"
}

        上述通过for-in语句遍历映射的运行效果如图2-10所示,可见映射内部每个元素的键名与键值都被获取到了。

2. 迭代器遍历

        映射的迭代器通过next函数得到下一个元素,接着需访问该元素的key属性获取键名,访问该元素的value属性获取键值。下面是在映射中运用迭代器遍历的代码例子:

btn_map_iterator.setOnClickListener {
    var desc = ""
    val iterator = goodsMutMap.iterator()
    //如果迭代器还存在下一个节点,就继续取出下一个节点的记录
    while (iterator.hasNext()) {
        val item = iterator.next()
        desc = "${desc}厂家:${item.key},名称:${item.value}\n"
    }
    tv_map_result.text = "手机畅销榜包含以下${goodsMutMap.size}款手机:\n$desc"
}

3. forEach遍历

        映射的forEach方法内部依旧采用匿名函数的形式,同时把元素的key和value作为匿名函数的输入参数。不过映射的forEach函数需要API 24及以上版本支持,开发时注意修改编译配置。下面是在映射中运用forEach遍历的代码例子:

btn_map_foreach.setOnClickListener {
    var desc = ""
    //映射的forEach函数需要API 24及以上版本支持
    //forEach内部使用key指代每条记录的键,使用value指代每条记录的值
    goodsMap.forEach { key, value -> desc = "${desc}厂家:${key},名称:${value}\n" }
    tv_map_result.text = "手机畅销榜包含以下${goodsMutMap.size}款手机:\n$desc"
    //tv_map_result.text = "Map的forEach函数需要API 24及以上版本支持"
}

2.5 小结

        本章介绍了Kotlin开发涉及的几种常见数据类型,包括以整型、浮点型、布尔型、字符型为代表的基本数据类型,还有这些基本数据类型数组的概念和运用,以及字符串的各种常见用法,最后是几种容器的常见操作方式。

        通过本章的学习,读者应能掌握以下技能:

        (1)学会Kotlin对基本数据类型的变量定义以及变量之间的类型转换。

        (2)学会Kotlin对基本类型数组的声明方式以及数组变量的常见用法。

        (3)学会Kotlin对字符串的各种处理操作以及字符串模板的书写格式。

        (4)学会Kotlin对容器的声明方式及其增、删、改、查操作,包括集合、队列、映射三种基本容器。

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

推荐阅读更多精彩内容