我的编程空间,编程开发者的网络收藏夹
学习永远不晚

【Kotlin】标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

【Kotlin】标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )


Kotlin 语言中 , 在 Standard.kt 源码中 , 为所有类型定义了一批标准库函数 , 所有的 Kotlin 类型都可以调用这些函数 ;







Kotlin 标准库函数 中的 apply 函数 ,

该函数可以看作 实例对象 的 配置函数 ,

传入 T.() -> Unit 类型 Lambda 表达式 作为参数 ,

该实例对象默认为 Lambda 表达式中的 this 参数 ;


apply 函数 的返回值 是 接收者对象 ,

也就是 调用 apply 函数 的实例对象 ,

同时也是 Lambda 表达式参数中的 this 参数 ;


apply 标准库函数原型 :

@kotlin.internal.InlineOnlypublic inline fun  T.apply(block: T.() -> Unit): T {    contract {        callsInPlace(block, InvocationKind.EXACTLY_ONCE)    }    block()    return this}

代码示例 : 创建 File 文件对象 , 并为该文件设置 可读 , 可写 , 可执行 权限 ;

import java.io.Filefun main() {    val file = File("hello.txt")    file.setReadable(true)    file.setWritable(true)    file.setExecutable(true)}

apply 函数代码示例 : 后面设置 可读 , 可写 , 可执行 权限的配置操作 , 可以在 apply 标准库函数中完成 , 代码如下 :

import java.io.Filefun main() {    val file = File("hello.txt").apply {        this.setReadable(true)        this.setWritable(true)        this.setExecutable(true)    }}






Kotlin 标准库函数 中的 let 函数 ,

可以传入 (T) -> R 类型 Lambda 表达式 作为参数 ,

该 匿名函数 中 使用 it 默认变量 获取 调用者 实例对象 ;


apply 函数与 let 函数的区别 :

  • apply 函数的 返回值是 调用者 ;
  • let 函数的 返回值是 Lambda 表达式的最后一行 ;

let 函数原型 :

@kotlin.internal.InlineOnlypublic inline fun  T.let(block: (T) -> R): R {    contract {        callsInPlace(block, InvocationKind.EXACTLY_ONCE)    }    return block(this)}

代码示例 :

fun main() {    val name = "tom".let {        it.capitalize()    }    println(name)}

上述代码中 , 调用 字符串 “tom” let 函数 ,

在 let 函数中 , 将首字母变为大写 , 并返回 ,

let 函数返回的是 匿名函数 的最后一行 , 因此将 “Tom” 字符串 返回了 ;

如果将 let 函数换成 apply 函数 , 则返回的就是 “tom” 字符串本身 , 不是 Lambda 表达式的最后一行 ;

执行结果 :

Tom

在这里插入图片描述

let 函数与 空安全操作符 ?. , 空合并操作符 ?: 结合使用 , 可替代 if 语句效果 ;

代码示例 :

fun main() {    val name: String? = null    println(getName(name))}fun getName(name: String?): String {    return name?.let {        "欢迎 $name 同学"    }?: "name 为空"}

在上述函数中 , 首先确定 name 变量是否为空 ,

如果 name 为空 , 则 name?.let {...} 为空 , 后面的 let 函数根本不会执行 ,

此时会取 空合并操作符 ?: 后面的值作为整个表达式的值 ;

如果 name 不为空 , 则 执行 let 函数 , 整个表达式的值 就是 let 函数的最后一行 返回值 ;

执行结果 :

name 为空

在这里插入图片描述







1、run 函数传入 Lambda 表达式作为参数


run 标准库函数原型如下 :

@kotlin.internal.InlineOnlypublic inline fun  T.run(block: T.() -> R): R {    contract {        callsInPlace(block, InvocationKind.EXACTLY_ONCE)    }    return block()}

run 函数 传入 T.() -> R 类型 Lambda 表达式 作为参数 ,

该 run 函数的 返回值 就是 Lambda 表达式 的返回值 ;


代码示例 : 在下面的代码中 ,

run 函数的 Lambda 表达式参数 返回的是 boolean 类型的 true 值 ,

该值就是最终 run 函数的返回值 ;

fun main() {    val ret = "Hello".run {        true    }    println(ret)}

执行结果 :

true

在这里插入图片描述


2、run 函数传入函数引用作为参数


在上述函数原型中 :

public inline fun  T.run(block: T.() -> R): R {}

run 函数 , 传入 T.() -> R 类型 函数参数 ,

此处也可以传入 函数引用 ;


利用 run 函数的该用法 , 可以进行链式调用 ;


代码示例 : 在下面的代码中 ,

"hello".run(::hasO) 代码 等价于 hasO("hello") 代码 ;

"hello".run(::hasO).run(::log) 代码 等价于 log(hasO("hello")) 代码 ;

"hello".run(::hasO).run(::log).run(::println) 代码 等价于 println(log(hasO("hello"))) 代码 ;

前者是链式调用代码 , 后者是正常的函数调用方式 ;

fun main() {    "hello"        .run(::hasO)        .run(::log)        .run(::println)}fun hasO(name: String): Boolean {    return name.contains("o")}fun log(hasO: Boolean): String {    if (hasO) {        return "name has o"    } else {        return "name doesn't has o"    }}

执行结果 :

name has o

在这里插入图片描述







with 函数run 函数 功能是一样的 ,

其使用形式不同 , with 函数是 独立使用的 ,

调用时 , 需要 将 接收者实例对象 作为 with 函数的 参数 ;


with 函数原型 :

@kotlin.internal.InlineOnlypublic inline fun  with(receiver: T, block: T.() -> R): R {    contract {        callsInPlace(block, InvocationKind.EXACTLY_ONCE)    }    return receiver.block()}

with 函数的第一个参数是 receiver: T 接收者 ,

第二个参数是 block: T.() -> R , 是 T.() -> R 类型的 Lambda 表达式 ;


代码示例 :

fun main() {    val str = with("hello") {        capitalize()    }    println(str)}

执行结果 :

Hello

在这里插入图片描述


上述 with 函数的执行效果与下面的 run 函数执行效果是相同的 ;

代码示例 :

fun main() {    val str = "hello".run {        capitalize()    }    println(str)}

执行结果 :

Hello

在这里插入图片描述







also 函数 功能与 let 函数 功能 类似 ;

also 函数 将 接收者 ( 函数调用者 ) 作为参数传递给 Lambda 表达式参数 ,

并返回 接收者实例对象本身 ;


also 函数 与 let 函数 返回值不同 ,

also 函数 返回 接收者对象本身 ,

let 函数 返回 Lambda 表达式的最后一行 ;


also 函数 返回 接收者对象本身 , 那么就可以使用该特性 , 对 接收者 执行 函数式编程的 链式调用 ;


代码示例 :

fun main() {    val str = "hello".also {        println(it)    }.also {        // 该对象的生命周期仅限于该闭包        println(it.capitalize())    }    // 最终打印的是最初的 接收者对象    println(str)}

执行结果 :

helloHellohello

在这里插入图片描述







takeIf 函数 的 返回值 由其 Lambda 表达式参数的返回值 确定 ,

Lambda 表达式 返回 true , 则 返回 接收者对象 ;

Lambda 表达式 返回 false , 则 返回 null 空值 ;


takeIf 函数 的功能 也可以使用 if 语句实现 ,

但是该函数 可以 直接 作用于 接收者对象 , 非常适合进行 函数式编程 的 链式调用 场景 ,

如果使用 if 语句 , 需要分 多行代码实现 , 还要定义临时变量 ;


takeIf 函数原型 :

@kotlin.internal.InlineOnly@SinceKotlin("1.1")public inline fun  T.takeIf(predicate: (T) -> Boolean): T? {    contract {        callsInPlace(predicate, InvocationKind.EXACTLY_ONCE)    }    return if (predicate(this)) this else null}

代码示例 : 在下面的代码中 ,

takeIf 函数 的 Lambda 表达式参数中 ,

使用 it.contains("o") 判断 接收者 字符串中是否包含 "o" 字符串 ,

如果返回 true , 则返回 接收者本身 , 否则返回 null ;

fun main() {    val str = "hello".takeIf {        it.contains("o")    }?.capitalize()    println(str)}

执行结果 :

Hello

在这里插入图片描述







takeUnless 函数 与 takeIf 函数 效果正好相反 ;


takeUnless 函数 的 返回值 由其 Lambda 表达式参数的返回值 确定 ,

Lambda 表达式 返回 false , 则 返回 接收者对象 ;

Lambda 表达式 返回 true , 则 返回 null 空值 ;


takeUnless 函数原型 :

@kotlin.internal.InlineOnly@SinceKotlin("1.1")public inline fun  T.takeUnless(predicate: (T) -> Boolean): T? {    contract {        callsInPlace(predicate, InvocationKind.EXACTLY_ONCE)    }    return if (!predicate(this)) this else null}

代码示例 : 在下面的代码中

takeUnless 函数的 Lambda 表达式 参数 返回的结果 为 true ,

因此 "hello".takeUnless { it.contains("o") } 的返回值为 null ,

由于后面的 ?.capitalize() 是空安全操作符调用 , 接收者为空的情况下不执行 ,

最终的 str 值为 null ;

fun main() {    val str = "hello".takeUnless {        it.contains("o")    }?.capitalize()    println(str)}

执行结果 :

null

在这里插入图片描述

来源地址:https://blog.csdn.net/shulianghan/article/details/128640441

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

【Kotlin】标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

【Kotlin】标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 )

文章目录 一、apply 标准库函数二、let 标准库函数三、run 标准库函数1、run 函数传入 Lambda 表达式作为参数2、run 函数传入函数引用作为参数 四、with 标准库函数五、also 标准库函数六、take
2023-08-17

【Kotlin】Kotlin 函数总结 ( 具名函数 | 匿名函数 | Lambda 表达式 | 闭包 | 内联函数 | 函数引用 )

文章目录 一、函数头声明二、函数参数1、默认参数值2、具名参数 三、Unit 函数四、TODO 函数抛出异常返回 Nothing 类型五、反引号函数名六、匿名函数七、匿名函数的函数类型八、匿名函数的隐式返回九、匿名函数参数十、
2023-08-19

【Kotlin】函数式编程 ② ( 过滤函数 | predicate 谓词函数 | filter 过滤函数 | 合并函数 | zip 函数 | folder 函数 | 函数式编程意义 )

文章目录 一、过滤函数二、filter 函数原型三、filter 过滤函数代码示例1、filter 函数简单示例2、filter 过滤函数与 flatMap 变换函数 组合使用示例3、filter 过滤函数与 map 变换函数 组合
2023-08-19

【Kotlin】函数式编程 ① ( 函数式编程简介 | 高阶函数 | 函数类别 | Transform 变换函数 | 过滤函数 | 合并函数 | map 变换函数 | flatMap 变换函数 )

文章目录 一、函数式编程简介1、编程范式2、高阶函数3、函数式编程4、前端开发技术 二、函数类别三、变换函数四、map 变换函数1、map 函数原型分析2、map 函数设计理念3、代码示例 五、flatMap 变换函数1、f
2023-08-19

Kotlin示例讲解标准函数with与run和apply的使用

Kotlin的标准函数是指Standard.kt文件中定义的函数,任何Kotlin代码都可以自由地调用所有的标准函数。文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
2022-11-13

【Kotlin】DSL 领域特定语言 ( apply 标准库函数分析 | 普通匿名函数 | 扩展匿名函数 | 泛型扩展匿名函数 )

文章目录 一、DSL 领域特定语言二、apply 标准库函数分析1、apply 函数展示2、apply 函数原型分析函数原型参数和返回值分析 3、匿名函数类型分析4、扩展函数回顾5、泛型扩展函数函数类型6、泛型扩展匿名函数7、
2023-08-30

PHP 函数与 Kotlin 函数的区别?

php 与 kotlin 函数的区别:php 函数返回类型可选、按值传递参数、支持变量数量参数、可声明为静态函数、允许匿名函数;kotlin 函数返回类型明确、参数可按值或引用传递、不支持变量数量参数、仅为成员函数或顶级函数、只能使用 la
PHP 函数与 Kotlin 函数的区别?
2024-04-25

python apply()函数

>>> help(apply)Help on built-in function apply in module __builtin__:apply(...) apply(object[, args[, kwargs]]) -> va
2023-01-31

PHP 函数与 Kotlin 函数对比分析

php 和 kotlin 函数处理方式对比:声明:php 使用 function,kotlin 使用 fun。参数传递:php 按值传递,kotlin 可选按值或按引用。返回值:php 返回值或 null,kotlin 返回值或 unit(
PHP 函数与 Kotlin 函数对比分析
2024-04-24

Python 函数总结

声明和调用函数:    声明函数的方法是用def关键字,函数名及小括号里面的参数列表。def foo(x): print x    调用函数:给出函数名和一小对括号,并放入所需参数:#!/usr/bin/env pytho
2023-01-31

C语言标准库函数qsort(快速排序函数)

qsort函数是C语言标准库中提供的一个快速排序函数。它的函数原型如下:```cvoid qsort(void *base, size_t nmemb, size_t size,int (*compar)(const void *, con
2023-09-23

python中apply函数和apply_async函数有什么区别

这篇文章主要介绍“python中apply函数和apply_async函数有什么区别”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“python中apply函数和apply_async函数有什么区别”
2023-07-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录