Kotlin入门(6)条件分支的实现
上一篇文章介绍了字符串的相关操作,其中示例代码用到了if和for语句,表面上看,Kotlin对控制语句的处理与Java很像,可实际上,Kotlin在这方面做了不少的改进,所以本篇和下一篇文章就分别介绍Kotlin如何操作条件判断和循环语句。
说起条件判断,最简单的莫过于人尽皆知的if...else...了,这个东东从C语言延续到Java,再进化到Kotlin,基本用法仍是一样的,看看下面的示例代码就知道了:
var is_odd:Boolean = true; tv_puzzle.text = "凉风有信,秋月无边。打二字" btn_if_simple.setOnClickListener { if (is_odd == true) { tv_answer.text = "凉风有信的谜底是“讽”" } else { tv_answer.text = "秋月无边的谜底是“二”" } is_odd = !is_odd }
以上代码的作用是,奇数次点击按钮,界面展示凉风有信的谜底;偶数次点击按钮,界面展示秋月无边的谜底。看似不能再简单的判断语句,谁能料到Kotlin也要加以简化?注意到两个谜底都是显示在控件tv_answer上,所以两个分支都出现了“tv_answer.text =”的语句。Kotin在这里要做的优化,便是允许分支语句返回字符串,从而在条件语句外层直接对tv_answer赋值,优化后的代码如下所示:
btn_if_simple.setOnClickListener { tv_answer.text = if (is_odd == true) { "凉风有信的谜底是“讽”" } else { "秋月无边的谜底是“二”" } is_odd = !is_odd }
以上的优化代码可以进一步改进,因为每个分支内部只有一个字符串返回值,所以不妨去掉大括号,并且把整个条件语句精简到一行代码,就像下面这样:
btn_if_value.setOnClickListener { tv_answer.text = if (is_odd==true) "凉风有信的谜底是“讽”" else "秋月无边的谜底是“二”" is_odd = !is_odd }
精简了的代码会不会似曾相识?仿佛脱胎于C语言跟Java的三元运算符“变量名=条件语句?取值A:取值B”。可是Kotlin并不提供这个三元运算符,因为使用上述的if/else语句已经实现了同样的功能,所以多余的三元运算符就被取消了。
三元运算符既已被取消,一旁的switch/case瑟瑟发抖,嘴里嘟囔道:“俺这个多路分支还在不在呀?”看官莫急,虽然Kotlin对if/else进行了增强,但是仍无法取代switch/case;相反的是,Kotlin对多路分支的功能做了大幅扩充,当然由于原来的switch/case机制存在局限,故而Kotlin推出了新的关键字,即when/else来处理多路分支的条件判断。
下面来个多路分支里的when/else语句的具体代码:
var count:Int = 0 btn_when_simple.setOnClickListener { when (count) { 0 -> tv_answer.text = "凉风有信的谜底是“讽”" 1 -> tv_answer.text = "秋月无边的谜底是“二”" //if语句可以没有else,但是when语句必须带上else else -> tv_answer.text = "好诗,这真是一首好诗" } count = (count+1) % 3 }
从以上代码可以看出when/else与switch/case有以下几点区别:
1、关键字switch被when取代:
2、判断语句“case 常量值:”被“常量值 ->”取代;
3、每个分支后面的break语句取消了,Kotlin默认一个分支处理完就直接跳出多路语句;
4、关键字default被else取代;
跟优化后的if/else一样,Kotlin中的when/else也允许有返回值,所以上面多路分支代码可优化为如下代码:
btn_when_value.setOnClickListener { tv_answer.text = when (count) { 0 -> "凉风有信的谜底是“讽”" 1 -> "秋月无边的谜底是“二”" else -> "好诗,这真是一首好诗" } count = (count+1) % 3 }
以往Java在使用switch/case时有个限制,就是case后面只能跟常量,不能跟变量,否则编译不通过。现在Kotlin去掉了这个限制,进行分支处理时允许引入变量判断,当然引入具体的运算表达式也是可以的。引入变量判断的演示代码如下:
var odd:Int = 0 var even:Int = 1 btn_when_variable.setOnClickListener { tv_answer.text = when (count) { odd -> "凉风有信的谜底是“讽”" even -> "秋月无边的谜底是“二”" else -> "好诗,这真是一首好诗" } count = (count+1) % 3 }
引入变量判断只是Kotlin牛刀小试,真正的功能扩充还在后面。原来的switch/case机制,每个case仅仅对应一个常量值,如果五个常量值都要进入某个分支的话,只能并列写五个canse语句,然后才跟上具体的分支处理语句。在when/else机制中便无需如此麻烦了,这五个常量值并排在一起用逗号隔开即可;如果几个常量值刚好是连续数字,可以使用“in 开始值..结束值”指定区间范围;举一反三,如果要求不在某个区间范围,则使用语句“!in 开始值..结束值”。扩展功能后的代码举例如下:
btn_when_region.setOnClickListener { tv_answer.text = when (count) { 1,3,5,7,9 -> "凉风有信的谜底是“讽”" in 13..19 -> "秋月无边的谜底是“二”" !in 6..10 -> "当里的当,客官你来猜猜" else -> "好诗,这真是一首好诗" } count = (count+1) % 20 }
多路分支的故事还没完,Kotlin设定了when/else语句不仅仅判断变量值,也可以判断变量的类型,如同Java的关键字instanceof那样。比如Java代码若想知晓某个变量是否为字符串类型,则使用以下代码进行判断:
if (str instanceof String) { ... }
那么在Kotlin中,类型判断一样沿用when/else语句,只不过在分支判断时采取“is 变量类型 ->”这种形式。下面是演示类型判断的代码,在变量countType为Long、Double、Float三种类型时做多路处理:
var countType:Number; btn_when_instance.setOnClickListener { count = (count+1) % 3 countType = when (count) { 0 -> count.toLong(); 1 -> count.toDouble() else -> count.toFloat() } tv_answer.text = when (countType) { is Long -> "此恨绵绵无绝期" is Double -> "树上的鸟儿成双对" else -> "门泊东吴万里船" } }
总结一下,对于条件分支的处理,Kotlin实现了简单分支和多路分支,其中简单分支跟Java一样都是if/else,多路分支则由Java的switch/case升级为when/else。同时,Kotlin的条件分支允许有返回值,可算是一大改进。另外,Java的三元运算符“变量名=条件语句?取值A:取值B”,在Kotlin中取消了,对应功能改为使用if/else实现;Java的关键字instanceof也取消了,对应的类型判断功能被纳入到when/else机制中。
__________________________________________________________________________
本文现已同步发布到微信公众号“老欧说安卓”,打开微信扫一扫下面的二维码,或者直接搜索公众号“老欧说安卓”添加关注,更快更方便地阅读技术干货。