Kotlin常用函数let,with,run,apply用法与区别案例详解

在kotlin编程中let、with、run、apply这些函数使用率是非常高的,有时候可以通用,差别很小,但如果能记住他们的不同点,可以更加合理的选择使用。

在这之前首先要了解一下Lambda表达式的一些规则,这会帮助你理解使用这些函数的时候有没有( )可不可以用it代替参数等。因为这些函数的最后一个参数都是lambda。

如何理解lambda呢?可以把lambda理解为就是一个对象,但这个对象比较特殊,它是一段代码,既然是对象就可以作为函数的参数使用。这种对象称为函数对象。
lambda表达式的语法格式举例:
{ x: Int, y: Int -> x + y }
lambda 表达式是括在“{ }”中, “ ->” 前面为lambda参数,后面为函数体。为了方便理解用普通函数推导一下变化过程:

fun sum(x: Int, y: Int) = x + y

首先去掉了关键字fun和方法名:

(x: Int, y: Int) = x + y

之后分为了2部分,“=” 前面作为lambda参数,后面作为函数体,中间用“->” 代替:

(x: Int, y: Int) ->  x + y

最后去掉参数的小括号,在外面加上大括号:

{x: Int, y: Int ->  x + y}

最终得到的lambda表达式就可以作为参数使用了:

fun useLambda(a: Int, {x: Int, y: Int ->  x + y}){
  ...代码省略...
}

了解了基础的语法格式后,在来看下,lambda的几个重要规则:
Kotlin规则1:如果函数的最后一个参数是lambda,lambda 表达式可以放在圆括号之外(拖尾 lambda):

val result = useLambda(a: Int){x: Int, y: Int ->  x + y}

Kotlin规则2:如果函数只有lambda一个参数时候,圆括号可以省略:

修改了useLambda函数:
fun useLambda({str: String ->  str.length}){
  ...代码省略...
}
规则1改为:
val result = useLambda(){str: String ->  str.length}
规则2改为:
val result = useLambda{str: String ->  str.length}

Kotlin规则3:如果lambda表达式只有一个参数时候,不用声明参数,可以用隐式名称it代替参数:

val result = useLambda{it.length}

Kotlin规则4:lambda表达式中有参数未使用的时候,可用下划线“_” 代替

useLambda(){_, y ->  y * y}

了解了lambda表达式之后再来分析函数
let函数用法:

val result = obj.let {
   //TODO 使用it访问obj对象的方法或者属性
   it.xxx()
   //TODO 最后一行代码的结果为返回值赋值给result
}

从let函数写法和lambda的规则可以看出,它只有一个lambda参数,并且这个lambda的参数也只有一个,就是调用者对象本身。let最常用于判断对象空指针之后有连续操作的场景。举例:

obj?.let{
   it.dosomething1()
   it.dosomething2()
   it.dosomething3()
}

with函数用法:

val result = with(obj) {
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
//TODO 最后一行代码的结果为返回值赋值给result
}

同样我们可以看出,with函数有2个参数,第一个参数为要处理的对象,最后一个是lambda参数,根据规则1,放在了圆括号后边。在这个lambda表达式中提供了第一个参数的对象可以用this指定,也可以省略。

run函数用法:

val result = obj.run{
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
//TODO 最后一行代码的结果为返回值赋值给result
}

run函数只有一个lambda参数,run看上去是let与with的结合。因为run可以解决判空调用还可以去掉“it”解决let函数隐式名称调用问题。

apply函数用法:

val result = obj.apply{
//TODO 直接访问obj对象的方法或者属性,或者使用this.调用
}

可以看出与run函数几乎一样,唯一的区别是lambda中没有指定返回值。返回的是调用者对象本身,也就是obj。

总结:

函数 参数个数 内部访问上下文对象 返回值
let 1 it lambda最后一行代码返回值
with 2 this 或者 省略 lambda最后一行代码返回值
run 1 this 或者 省略 lambda最后一行代码返回值
apply 1 this 或者 省略 返回调用者对象

到此这篇关于Kotlin常用函数let,with,run,apply用法与区别案例详解的文章就介绍到这了,更多相关Kotlin常用函数let,with,run,apply用法与区别内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Kotlin修饰符lateinit(延迟初始化)案例详解

    Kotlin定义变量一般有如下写法 lateinit var name: String var age: String? = null 那么用lateinit 修饰和下面那种有什么区别呢,我们来看一下这两行代码反编译成java代码是什么样子的. @NotNull public String name; @Nullable private String age; @NotNull public final String getName() { String var10000 = this.name

  • Kotlin lateinit与by lazy案例详解

    lateinit 和 lazy 是 Kotlin 中的两种不同的延迟初始化的实现 lateinit 只用于变量 var,而 lazy 只用于常量 val lazy 应用于单例模式(if-null-then-init-else-return),而且当且仅当变量被第一次调用的时候,委托方法才会执行. lazy()是接受一个 lambda 并返回一个 Lazy <T> 实例的函数,返回的实例可以作为实现延迟属性的委托: 第一次调用 get() 会执行已传递给 lazy() 的 lambda 表达式并

  • Kotlin如何安全访问lateinit变量的实现

    Kotlin设计之初就是不允许非null变量在声明期间不进行初始化的,为了解决这个问题,Kotlin lateinit 允许我们先声明一个变量,然后在程序执行周期的将来某个时候将其初始化,让编译检查时不会 因为属性变量未被初始化而报错.如果未初始化将导致以下异常: kotlin.UninitializedPropertyAccessException: lateinit property mList has not been initialized 所以我们在 Kotlin 1.2及更高版本上,

  • kotlin之协程的理解与使用详解

    前言         为什么在kotlin要使用协程呢,这好比去了重庆不吃火锅一样的道理.协程的概念并不陌生,在python也有提及.任何事务的作用大多是对于所依赖的环境相应而生的,协程对于kotlin这门语言也不例外.协程的优点,总的来说有如下几点:轻量级,占用更少的系统资源: 更高的执行效率: 挂起函数较于实现Runnable或Callable接口更加方便可控: kotlin.coroutine 核心库的支持,让编写异步代码更加简单.当然在一些不适应它的用法下以上优势也会成为劣势. 1.协程

  • Kotlin常用函数let,with,run,apply用法与区别案例详解

    在kotlin编程中let.with.run.apply这些函数使用率是非常高的,有时候可以通用,差别很小,但如果能记住他们的不同点,可以更加合理的选择使用. 在这之前首先要了解一下Lambda表达式的一些规则,这会帮助你理解使用这些函数的时候有没有( )可不可以用it代替参数等.因为这些函数的最后一个参数都是lambda. 如何理解lambda呢?可以把lambda理解为就是一个对象,但这个对象比较特殊,它是一段代码,既然是对象就可以作为函数的参数使用.这种对象称为函数对象. lambda表达

  • JavaScript函数之call、apply以及bind方法案例详解

    总结 1.相同点 都能够改变目标函数执行时内部 this 的指向 方法的第一个参数用于指定函数执行时内部的 this 值 支持向目标函数传递任意个参数 若不向方法的第一个参数传值或者传递 undefined.null,则在 JavaScript 正常模式下,目标函数内部的 this 指向 window 对象,严格模式下,分别指向 undefined.null. 2.区别 apply() 方法可接收两个参数,而 call() 和 bind() 方法则可接收多个参数. apply() 方法向目标函数

  • js中Map和Set的用法及区别实例详解

    目录 首先了解一下 Map 再来了解一下 Set 总结Map和Set的区别 结语: 首先了解一下 Map Map 是一组键值对的结构,和 JSON 对象类似. (1) Map数据结构如下 这里我们可以看到的是Map的数据结构是一个键值对的结构 (2) key 不仅可以是字符串还可以是对象 var obj ={name:"小如",age:9} let map = new Map() map.set(obj,"111") 打印结果如下 (3) Map常用语法如下 //初

  • PHP中isset、empty的用法与区别示例详解

    前言 在编写程序调用变量时,遇到未定义的变量时,会报错,这是就需要我们对变量先进行判断,再进行相关操作. 这里主要记录两个变量判断函数的使用:isset()与empty() 1.isset - 检测变量是否已设置并且非 NULL isset ( mixed $var [, mixed $... ] ) : bool 如果已经使用 unset() 释放了一个变量之后,它将不再是 isset(). 若使用 isset() 测试一个被设置成 NULL 的变量,将返回 FALSE. 同时要注意的是 nu

  • Python函数any()和all()的用法及区别介绍

    引子 平常的文本处理工作中,我经常会遇到这么一种情况:用python判断一个string是否包含一个list里的元素. 这时候使用python的内置函数any()会非常的简洁: fruits = ['apple', 'orange', 'peach'] str = "I want some apples" if any(element in str for element in fruits): print "string contains some fruits."

  • SQL之patindex函数的用法案例详解

    语法格式:PATINDEX ( '%pattern%' , expression ) 返回pattern字符串在表达式expression里第一次出现的位置,起始值从1开始算. pattern字符串在expression表达式里没找就返回0,对所有有效的文本和字符串就是有效的数据类型. 描述一下此函数的具体用法: 1. PATINDEX ( '%pattern%' , expression ) '%pattern%'的用法类似于 like '%pattern%'的用法,也就是模糊查找其patte

  • C++ seekg函数用法案例详解

    C++ seekg函数用法详解 很多时候用户可能会这样操作,打开一个文件,处理其中的所有数据,然后将文件倒回到开头,再次对它进行处理,但是这可能有点不同.例如,用户可能会要求程序在数据库中搜索某种类型的所有记录,当这些记录被找到时,用户又可能希望在数据库中搜索其他类型的所有记录. 文件流类提供了许多不同的成员函数,可以用来在文件中移动.其中的一个方法如下: seekg(offset, place); 这个输入流类的成员函数的名字 seekg 由两部分组成.首先是 seek(寻找)到文件中的某个地

  • StretchBlt函数和BitBlt函数用法案例详解

    StretchBlt和BitBlt都用在双缓冲视图中,用来显示一幅图像 一.StretchBlt 函数从源矩形中复制一个位图到目标矩形,必要时按目标设备设置的模式进行图像的拉伸或压缩.也即是将内存中的位图拷贝到屏幕上,并且可以根据屏幕画图区的大小来进行伸缩,适应响应的屏幕(或图像控件) BOOL StretchBlt( int x, int y, int nWidth, int nHeight, CDC* pSrcDC, int xSrc, int ySrc, int nSrcWidth, in

  • Python threading Local()函数用法案例详解

    目录 前言 local() 函数是什么? local()函数如何用? 1. 不做标记,不做隔离 2.使用local()函数加以控制 3. 模拟实现local()的功能,创建一个箱子 4. 简化代码操作,进一步模拟实现local()函数 总结 前言 当多线程访问同一个公共资源时,如果涉及到修改该公共资源的操作就可能会出现由于数据不同步导致的线程安全问题.一般情况下我们可以通过给公共资源加互斥锁的方式来处理该问题. 当然,除非必须将多线程使用的资源设置为公共资源的情况.如果一个资源不需要在多个线程之

  • C++ assert()函数用法案例详解

    1. 简介 assert宏的原型定义在<assert.h>中,其作用是如果它的条件返回错误,则终止程序执行. 原型定义: #include <assert.h> void assert( int expression ); assert的作用是先计算表达式 expression ,如果其值为假(即为0),那么它先向stderr打印一条出错信息,然后通过调用 abort 来终止程序运行.请看下面的程序清单badptr.c: #include <stdio.h> #incl

随机推荐