Kotlin中常见内联扩展函数的使用方法教程
作者:Knight_Davion 发布时间:2023-07-04 13:46:12
前言
Kotlin一个强大之处就在于它的扩展函数,巧妙的运用这些扩展函数可以让你写出的代码更加优雅,阅读起来更加流畅,下面总结了在开发中经常用到的一些内联扩展函数。经常有小伙伴搞不懂with,run,apply等等这些函数该怎么用,在哪里用,我的建议是先记住每个函数的功能(无非就是它需要什么参数?返回值是什么?)记住这两点再根据实际开发中的场景慢慢的就能熟练运用了。其实这些函数极其类似,不同的函数可以完成同样的功能,通过下面的实例也能看出。而在我以往的开发经验中这些函数主要的使用场景有两个,一是非空判断,二是对象的初始化或者本身及方法的频繁调用。
内联和正常函数的区别不在于定义函数的异同点。定义的时候只需要加一个标识,就可以让正常函数变为内联函数。实际两者的区别是在实际执行时的处理机制上。内联是耗用性能低,比正常函数少了压栈和出栈的操作,是一种以空间换时间的方式。当函数体少,以及被频繁调用的函数才适合被定义为内联函数
1. with
定义:fun <T, R> with(receiver: T, block: T.() -> R): R
功能:将对象作为函数的参数,在函数内可以通过 this指代该对象。返回值为函数的最后一行或return表达式。
实例:
1.在自定义view中当我们初始化画笔时很多时候我们会写下边的代码
var paint = Paint()
paint.color = Color.BLACK
paint.strokeWidth = 1.0f
paint.textSize = 18.0f
paint.isAntiAlias = true
如果使用with,那么就可以写成这样
var paint = Paint()
with(paint) {
color = Color.BLACK
strokeWidth = 1.0f
textSize = 18.0f
isAntiAlias = true
}
省去了paint.后书写起来感觉会更加自然
2.在声明一些集合的场景比如:
var list= mutableListOf<String>()
list.add("1")
list.add("2")
list.add("3")
使用with可以写成
var list = with(mutableListOf<String>()) {
add("1")
add("2")
add("3")
this
}
开发中还有很多场景可以使用with功能,理解了with的功能也就能够灵活运用了。
2. takeIf和takeUnless
takeif
定义:fun <T> T.takeIf(predicate: (T) -> Boolean): T?
功能:传递一个函数参数,如果函数结果为true,返回T对象,否则返回null。
实例:使用File文件时通常会判断file是否存在,比如
var file = File("filePath")
if (file.exists()) {
//do something
} else {
return false
}
使用takeif后
var file = File("filePath").takeIf { it.exists() }?:return false
//do something
takeUnless
定义:fun <T> T.takeUnless(predicate: (T) -> Boolean): T?
功能:与takeIf相反,参数函数返回false时返回T对象,否则返回null,这里不再举例。
3. run
定义:
(1)fun <R> run(block: () -> R): R
(2)fun <T, R> T.run(block: T.() -> R): R
功能:调用run函数返回值为函数体最后一行,或return表达式。
实例:
返回最后一行
kotlin.run {
println("11")
println("22")
}
结果:
I/System.out: 11
I/System.out: 22
返回return表达式,return后面的代码不再执行(注意写法@run)
kotlin.run {
return@run println("11")
println("22")
}
结果:
I/System.out: 11
4. repeat
定义:fun repeat(times: Int, action: (Int) -> Unit)
功能:重复执行action函数times次,times从0开始
实例:与for循环功能类似,例如
repeat(5){ println("count:$it") }
等价于
for (i in 0..4) { println("count:$i") }
或者
(0..4).forEach{println("count:$it")}
5. let
定义:fun <T, R> T.let(block: (T) -> R): R
功能:调用对象(T)的let函数,则该对象为函数的参数。在函数内可以通过 it 指代该对象。返回值为函数的最后一行或指定return表达式。
实例:有点类似于run(),let在使用中可用于空安全验证,变量?.let{}
例如
val data = ……
data?.let {
…… // 假如data不为null,代码会执行到此处
}
6. apply
定义:fun <T> T.apply(block: T.() -> Unit): T
功能:调用对象的apply函数,在函数范围内,可以任意调用该对象的任意方法,并返回该对象。
实例:
var list = mutableListOf<Int>().apply {
add(1)
add(2)
add(3)
}
注意:他和run函数的区别,run返回的是最后一行,apply返回的是对象本身,由apply函数的定义我们可以看出apply适用于那些对象初始化需要给其属性赋值的情况。
还是用画笔的例子
原始的
var paint = Paint()
paint.textSize = 14.0f
paint.color = Color.WHITE
paint.isAntiAlias = false
使用apply后
var paint = Paint().apply {
textSize = 14.0f
color = Color.WHITE
isAntiAlias = false
}
此外由于apply函数返回的是其对象本身,那么可以配合?.完成多级的非空判断操作,或者用于建造者模式的Builder中
7. also
定义:fun <T> T.also(block: (T) -> Unit): T
功能:调用对象的also函数,在函数块内可以通过 it
指代该对象,返回值为该对象本身。(注意其和let函数的区别,let返回的是最后一行,also返回的是对象本身)
实例:需要返回对象本身(this)的情况下,例如建造者模式。
来源:https://www.jianshu.com/p/d5f26907200d


猜你喜欢
- 1 . pom.xml添加相关依赖<parent> <groupId>org.spring
- 查询文档 & 基本操作为了方便学习, 本节中所有示例沿用上节的索引按照ID单个GET class_1/_doc/1查询结果:{ &n
- 1 前言敏感词过滤就是你在项目中输入某些字(比如输入xxoo相关的文字时)时要能检测出来,很多项目中都会有一个敏感词管理模块,在敏感词管理模
- 近期,Google宣布Kotlin成为了Android一级开发语言。于是就刚刚简单的研究了一下,查资料的时候发现现成的资料还是很少的,于是决
- 大家都知道 Android 的 Activity 是存着历史栈的,比如从 A -> B -> C,C 完成 finish 后回到
- OOP语言的三大特征即:面向对象的三个比较重要的思想封装官话:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口进
- 第一:写Cookies Response.Cookies["UserName"].Value="Guest&q
- 这是android新推出的一个,让卡片带立体感的一个控件,就是一个卡牌,有点类似于布局那种的东西,里面可以添加控件内容先看看运行的效果图:1
- 前言上篇博客我们介绍了如何创建一个spring项目,并且如何的存、取对象,介绍了相关方法,那么本篇博客将接着上篇博客的内容介绍如何更加简单的
- 抽象方法与虚方法的区别先说两者最大的区别:抽象方法是需要子类去实现的。虚方法是已经实现了的,可以被子类覆盖,也可以不覆盖,取决于需求。因为抽
- 通过VideoView播放视频的步骤:1、在界面布局文件中定义VideoView组件,或在程序中创建VideoView组件2、调用Video
- 摘要 2021年了,还有不支持弹幕的视频网站吗,现在各种弹
- 1. 源码阅读环境搭建ide:IntelliJ IDEA 2020.1包管理:gradleeureka版本:1.10.11Spring Cl
- 示例代码如下:launch(Dispatchers.Main) { // 第一部分 fl
- 本文通过一个简单的小例子简述SharpZipLib压缩文件的常规用法,仅供学习分享使用,如有不足之处,还请指正。什么是SharpZipLib
- 本文实例形式详述了Java实现一个程序运行时的启动窗口效果,如常用的Microsoft Word、 Borland JBuilder 等,这
- Redis不仅可作为缓存服务器,还可以用作消息队列。它的列表类型天生支持用作消息队列。如下图所示:由于Redis的列表是使用双向链表实现的,
- 概述对象实例由对象头、实例数据组成,其中对象头包括markword和类型指针,如果是数组,还包括数组长度;| 类型 | 32位JVM | 6
- 概述本文的编写初衷,是想了解一下Spring Boot2中,具体是怎么序列化和反序列化JSR 310日期时间体系的,Spring MVC应用
- 使用idea的file-》settings-》plugins安装maven helper插件失败,安装页面总是提示installed,在in