带你踏入kotlin大门(四)|基本功_逻辑控制篇

简介: 本系列的上一篇文章中,笔者给大家讲述 kotlin 中的 头等公民 ——函数,相信 Kotlin 中函数的简单使用大家都很熟悉了,现在继续给大家讲述 Kotlin 中的逻辑控制语句。

前置知识

  • 已学习 Kotlin 变量、函数知识
  • Java 编程基础

前言

本系列的上一篇文章中,笔者给大家讲述 kotlin 中的 头等公民 ——函数,相信 Kotlin 中函数的简单使用大家都很熟悉了,现在继续给大家讲述 Kotlin 中的逻辑控制语句。

本篇主要讲述三大执行语句中的 条件判断语句循环语句,由于顺序执行语句与 Java 一样,没什么特点,我们前面文章中也多次使用过了,这里便不再赘述。

1.webp.jpg

条件判断语句

if 语句

Kotlinif 语句的实现和 java 中的 if 是没有太大区别的。我们需要关注的是其特性表达式化这些与 Java 不同的地方。

首先,我们依旧是用一段简单的 Java 代码来进行变化。

public int numMax(int num1, int num2){
    int res = 0;
    if(num1 > num2){
        res = num1;
    }else{
        res = num2;
    }
    return res;
}
复制代码

转化为 Kotlin 之后,是如下方的代码。这一段转换,使用到的是我们前文中使用到的一些内容,直接把函数给转化。

fun numMax(num1: Int, num2: Int): Int{
    var res = 0
    if(num1 > num2){
        res = num1
    }else{
        res = num2
    }
    return res
}
复制代码

但是上面这段代码的转化,显然是少一些 Kotlin 的特点的。什么特点呢?首先是 if 函数自身的特点。if 函数可以 表达式化/函数化 ,在这种情况下,可以直接把每个条件下的最后一条语句作为返回值。这是什么意思呢?具体看下面的代码变化。

fun numMax(num1: Int, num2: Int): Int{
    var res = if(num1 > num2){
        num1
    }else{
        num2
    }
    return res
}
复制代码

由上面的代码我们可以看到,当其 res 的赋值是表达式的时候,表达式中最后一行不能出现 res = xxx 这种类型的语句,因为 if 语句变为表达式的时候,是需要把最后一行作为结果返回的,所以最后一行需要是一个结果。

我们再次把它变化得更具 Kotlin 风格,这次是将 if 表达式直接作返回值返回。

fun numMax(num1: Int, num2: Int): Int{
    return = if(num1 > num2){
        num1
    }else{
        num2
    }
}
复制代码

同样的,既然函数体里面只返回一个 if 表达式,那么我们可以直接把 if 表达式继续向外提出,让整个函数体变为一个 if 表达式的结果。变化后的代码如下。

fun numMax(num1: Int, num2: Int) = if(num1 > num2) num1 else num2
复制代码

是的,这段代码最后变成了一行代码了,变成了纯表达式化的代码。这里要注意的是,把整个函数变成 if 表达式之后,对应的函数的声明也去掉了,这正对应我们上一篇文章所讲的知识点。

when 语句

Kotlin 中的 when 语句,大家初次听也许感觉很陌生。但是事实上,你可以将其看作是 Javaswitch 语句的加强版本。在 Kotlin 中,已不再有 switch 语句,全面使用 when 语句替代。when 语句即保留了 swich 语句的多重判定,同时又让其使用变得无比简洁且可接收各种类型的参数。所以,你也可以把 when 理解为多重 if 的简化版。

我们依旧以 Java 版本,先写出一段多重 if 语句代码。这里为什么不以 switch 举例?是因为 switch 语句直到在 jdk 1.7 以下只支持 int 类型变量,所以此处不以其作为例子。

下面的这段代码很简单,就是对传入的字符串作判断,根据不同的字符串来返回不同的整形数值。

public int checkString(String s){
    int sNum = 0;
    if(s.equals("a")){
        sNum = 1;
    }else if(s.equals("b")){
        sNum = 2;
    }else if(s.equals("c")){
        sNum = 3;
    }
    return sNum;
}
复制代码

将上述的代码转化为 Kotlin ,使用 when 语句来编写,就是如下的样式:

fun checkString(s: String) = when(s){
    "a" -> 1
    "b" -> 2
    "c" -> 3
    else -> 0
}
复制代码

是否顿时觉得变得无比明了简单呢!你会发现这种形式更加的适合人的思维。相比于 switch 和多重 if,这也让我们的代码编写过程变得更加简单。

从上面的示例代码,我们可以看出三点。

  1. when() 语句可以传入参数,事实上这个参数可以是任意类型的;
  2. when 下边的代码编写格式为:"匹配条件" -> {} ,这里没有了{}包含的函数体,是因为它是单行的,所以将其视为表达式,把 {} 省略了;
  3. when 语句需要返回值的时候,是以每个条件下的最后一行表达式作为返回值

但是 when功能远不止于此,让我们来看一下下面的代码。

fun checkInt(i: Any) = when(i){
    in 1..20 -> 1 //1
    is Long,Double -> 2 //2
    13 -> 3     //3
    else -> 0
}
复制代码

可以看到,上文中的每行的表达式和前面代码不一样了,其实这正体现了 when 广阔的判断功能。

上述的 注释1 处,判断 i 是否在 [1,20] 的区间内,其中 in 和 1..20 这两种用法,我在接下来的循环语句中会讲述含义和用法;

注释2 处,判断 i 是否为 Long 或者 Double 类型的,is 我们可以理解为 Java 中的 instanceof ,用于查看是否为对应子类的;

注释3 处,便是和前面一样,判断是否为13了。

1.webp.jpg

同时,当带有参数的功能不再满足于现有要求的时候,我们还可使用无参的 when 语句

fun checkString(s: String) = when{
    s.length == 2 -> 1
    s.startsWith("a") -> 2
    s == "c" -> 3
    else -> 0
}
复制代码

上述这段无参的 when 语句块,让我们的代码变得更加灵活,使得可以对各种外部的属性作判断,也可以对属性作各种 Boolean 类型的逻辑判断。所以说,when 更像是多重 if 的简约版本,功能强大的同时,使用也便捷。

1.webp.jpg

循环语句

Kotlin 中的循环语句,包括 whilefor 这两种循环语句。其中,while 语句和 Java 中的一样,倒是 for 语句在 Kotlin 中有很多不同,所以只在此介绍 for 语句。

for 语句

Kotlin 中的 for 语句,完全摒弃了传统的 for-i 循环,全面使用 for-each 循环。但是同时也对 for-each 循环做了一个拓展,让其变得简单易用。

在 Kotlin 中,我们可以将改进后的 for-each 循环称之为  for-in 循环。老规矩,我们先上一段 Java 代码,再对其进行转化。

private final List<ShoppingCartItem> items = new ArrayList<>();
public List<ShoppingCartItem> getItems() {
    List<ShoppingCartItem> copyItems = new ArrayList<>();;
    for (ShoppingCartItem i : items){//1
        copyItems.add(i.clone());
    }
    return Collections.unmodifiableList(copyItems);
}
复制代码

上面这段代码来自于我的 设计模式系列 中。可以看到上述的代码的注释1处使用了 for-each 。我们将其转化为 Kotlin 之后是这样子的。

private val items: MutableList<ShoppingCartItem> = ArrayList()
fun getItems(): List<ShoppingCartItem> {
    val copyItems: MutableList<ShoppingCartItem> = ArrayList()
    for (i in items) {//1
        copyItems.add(i.clone())
    }
    return Collections.unmodifiableList(copyItems)
}
复制代码

可以看到,最不一样的地方,就是  for-each 中的 :变成 in 了。这里是遍历集合,我们很好理解。

但是,要实现 Java 的 for-i 的用法该怎么办呢?在 Kotlin 中也有对应拓展的。

在介绍之前,我们先学习几个知识。

0..10 //代表的是 [0,10],左闭右闭
0 until 10 //代表的是 [0,10),左开右闭
复制代码

由此,我们可以写出 for-i 中1遍历到10的代码为:

for (i in 0..10) {           // Loop over a range from 0 to 10
    print(i)
}
复制代码

使用 until 关键字, for-i 中1遍历到10的代码可以写成:

for (i in 0 until 11) {           // Loop over a range from 0 to 10
    print(i)
}
复制代码

Kotlin 中的  for-in 默认是正序且步长为1的,我们可以分别使用 downTostep 两个关键字对其作更改 。

for (i in 10 downTo 1 step 2) {
    print(i)
}
复制代码

上述代码的运行结果为:

1.webp.jpg



相关文章
|
存储 Java 编译器
带你踏入kotlin大门(二)|基本功_变量篇
本篇正式开启 kotlin 的学习历程,带你正式踏入 kotlin 大门。
带你踏入kotlin大门(二)|基本功_变量篇
|
Java API Kotlin
带你踏入kotlin大门(三)|基本功_函数篇
本文给讲述 kotlin 中的 头等公民 ,函数。
带你踏入kotlin大门(三)|基本功_函数篇
|
设计模式 Java 开发者
带你踏入kotlin大门(五)|基本功_类和继承篇
本篇作为 Kotlin 基本功的倒数第二篇,将为大家带来关于 Kotlin 中类和继承的知识。如果你看到这篇文章的时候,还未听说过 Kotlin 或者从未学习过 Kotlin ,你可以点击链接从本系列的第一篇文章开始学习。
带你踏入kotlin大门(五)|基本功_类和继承篇
|
安全 Java 大数据
带你踏入Kotlin大门(一)|kotlin初识
作为一名 Java 使用者,相信大家对 Java 的各种规范已经深入了解,这些规范是一种行业标准,它使得 Java 具有了更好的兼容性。但随着函数式编程的兴盛,这些规范又开始束缚了开发者的创造,成为了拖累 Java 的因素。
|
设计模式 安全 Java
带你踏入Kotlin大门(六)|基本功_接口和特殊类篇
这篇文章是我们 Kotlin 基础章系列的最后一文了,前面我们利用了5篇文章讲解了 Java 开发者如何学好 Kotlin 的基本用法,每篇文章的篇幅都不长,可以在空余时间快速阅读,笔者希望学习完前五篇,再来学习这最后一篇文章,会更加的容易理解。
|
3月前
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
42 1
|
4月前
|
Android开发 开发者 Kotlin
告别AsyncTask:一招教你用Kotlin协程重构Android应用,流畅度飙升的秘密武器
【9月更文挑战第13天】随着Android应用复杂度的增加,有效管理异步任务成为关键。Kotlin协程提供了一种优雅的并发操作处理方式,使异步编程更简单直观。本文通过具体示例介绍如何使用Kotlin协程优化Android应用性能,包括网络数据加载和UI更新。首先需在`build.gradle`中添加coroutines依赖。接着,通过定义挂起函数执行网络请求,并在`ViewModel`中使用`viewModelScope`启动协程,结合`Dispatchers.Main`更新UI,避免内存泄漏。使用协程不仅简化代码,还提升了程序健壮性。
119 1
|
6月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
**Kotlin中的`by lazy`和`lateinit`都是延迟初始化技术。`by lazy`用于只读属性,线程安全,首次访问时初始化;`lateinit`用于可变属性,需手动初始化,非线程安全。`by lazy`支持线程安全模式选择,而`lateinit`适用于构造函数后初始化。选择依赖于属性特性和使用场景。**
187 5
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
|
5月前
|
调度 Android开发 开发者
【颠覆传统!】Kotlin协程魔法:解锁Android应用极速体验,带你领略多线程优化的无限魅力!
【8月更文挑战第12天】多线程对现代Android应用至关重要,能显著提升性能与体验。本文探讨Kotlin中的高效多线程实践。首先,理解主线程(UI线程)的角色,避免阻塞它。Kotlin协程作为轻量级线程,简化异步编程。示例展示了如何使用`kotlinx.coroutines`库创建协程,执行后台任务而不影响UI。此外,通过协程与Retrofit结合,实现了网络数据的异步加载,并安全地更新UI。协程不仅提高代码可读性,还能确保程序高效运行,不阻塞主线程,是构建高性能Android应用的关键。
67 4
|
6月前
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin中常见作用域函数
**Kotlin作用域函数概览**: `let`, `run`, `with`, `apply`, `also`. `let`安全调用并返回结果; `run`在上下文中执行代码并返回结果; `with`执行代码块,返回结果; `apply`配置对象后返回自身; `also`附加操作后返回自身
64 8