当前位置: 首页 > news >正文

购物网网站建设成都网站设计

购物网网站建设,成都网站设计,邵阳做网站公司,做网站体会noinline 顾名思义,noinline的意思就是不内联,这个关键字只能作用于内联高阶函数的某个函数类型的参数上,表明当前的函数参数不参与高阶函数的内联: inline fun fun1(doSomething1: () -> Unit, noinline doSomething2: () -&…

noinline

顾名思义,noinline的意思就是不内联,这个关键字只能作用于内联高阶函数的某个函数类型的参数上,表明当前的函数参数不参与高阶函数的内联:

inline fun fun1(doSomething1: () -> Unit, noinline doSomething2: () -> Unit) {Log.i("tag", "1")doSomething1()doSomething2()
}//调用
fun mainFun() {fun1({ Log.i("tag", "2") }, { Log.i("tag", "3") })
}//实际编译的代码
fun mainFun() {Log.i("tag", "1")Log.i("tag", "2")({Log.i("tag", "3")}).invoke()
}

但是这个关键字有什么用呢?

在kotlin中高阶函数的函数类型的参数我们可以直接当做一个函数去调用,但是函数类型的参数终究还是一个对象,既然是一个对象那么我们就可以以对象的形式去使用,就比如说作为函数返回值进行返回:

inline fun fun1(doSomething1: () -> Unit, doSomething2: () -> Unit): () -> Unit {Log.i("tag", "1")doSomething1()doSomething2()return doSomething2//这里编译器会提示报错,因为作为内联函数的函数类型参数,//它已经不能作为对象去使用了。如果不加inline关键字这里就是正确的
}

我们知道内联函数的函数类型参数是不会再创建函数对象的,也就是说它只是一个函数体而不是一个函数对象,所以它无法被当做一个对象那样作为返回值进行返回。

所以当我们在内联函数里面真的需要将一个函数类型的参数作为对象去使用时就需要noinline关键字去修饰它:

inline fun fun1(doSomething1: () -> Unit, noinline doSomething2: () -> Unit): () -> Unit {Log.i("tag", "1")doSomething1()doSomething2()return doSomething2//编译正常
}

crossinline

crossinline的含义字面上可以理解为对inline做局部加强内联

场景:

inline fun fun1(doSomething1: () -> Unit){Log.i("tag", "1")doSomething1()
}//调用
fun mainFun() {fun1 {Log.i("tag", "2")return
//按照一般的原则,这里的return结束的应该是fun1函数体中的逻辑,就是说Log.i("tag", "3")会被执行到。
//但是fun1作为内联函数会在编译时被完全铺平,这里return结束的就是最外面mainFun函数的逻辑,Log.i("tag", "3")就不会被执行到。}Log.i("tag", "3")
}//实际编译的代码
fun mainFun() {Log.i("tag", "1")Log.i("tag", "2")returnLog.i("tag", "3")
}

按照一般的原则,这里的return结束的应该是fun1函数体中的逻辑,就是说Log.i("tag", "3")会被执行到。

但是fun1作为内联函数会在编译时被完全铺平,这里return结束的就是最外面mainFun函数的逻辑,Log.i("tag", "3")就不会被执行到。

那我们在函数类型参数的lambda表达式中执行的return到底结束的是当前函数还是最外层的函数完全要看fun1到底是不是内联函数,这就让我们代码敲得很难受。

为此,kotlin提出了一个新规定:lambda表达式中不允许直接使用return,除非是内联函数的函数类型参数的lambda表达式,并且它结束的是最外层的函数逻辑。同时其他场景下的lambda表达式中可以通过return@label的形式来显示指定return结束的代码作用域 。 

fun fun1(doSomething1: () -> Unit){Log.i("tag", "1")doSomething1()
}fun mainFun() {fun1 {Log.i("tag", "2")//return//直接return在这里是不被允许的,因为fun1不是内联函数return@fun1//return@label的形式}Log.i("tag", "3")
}

再来看一种场景:

inline fun fun1(doSomething1: () -> Unit){Log.i("tag", "1")Runnable {doSomething1()//这里会编译报错}
}//调用
fun mainFun() {fun1 {Log.i("tag", "2")return}Log.i("tag", "3")
}

我们在内联函数fun1中将函数类型参数doSomething1放到了子线程里面去执行,这样doSomething1和fun1就属于间接调用的关系,

那mainFun函数中的return结束的到底是Runnable子线程中的逻辑还是mainFun函数中的逻辑呢?

所以kotlin是不允许直接这样写的,但是我确实有需求要这样在内联函数中间接调用函数类型的参数怎么办?

kotlin为此又新增了一条规定:内联函数中不允许对函数类型参数的间接调用,除非该函数类型参数被crossinline关键字修饰:(这就是局部加强内联的含义)

代码:

inline fun fun1(crossinline doSomething1: () -> Unit){Log.i("tag", "1")Runnable {doSomething1()}
}//调用
fun mainFun() {fun1 {Log.i("tag", "2")return//这里会编译报错}Log.i("tag", "3")
}

crossinline可以到达间接调用函数类型参数的目的 。

但是并没有解决“mainFun函数中的return结束的到底是Runnable子线程中的逻辑还是mainFun函数中的逻辑呢?

于是kotlin干脆在间接调用的情况下内联函数的函数类型参数的lambda表达式不允许直接使用return,即下面这两个规定不可以共存:

  1.  lambda表达式中不允许直接使用return,除非是内联函数的函数类型参数的lambda表达式,并且它结束的是最外层的函数逻辑
  2. 内联函数中不允许类似上述问题中对函数类型参数的间接调用,除非该函数类型参数被crossinline关键字修饰


最终代码:(用return@label的形式来显示指定return结束的代码作用域)

inline fun fun1(crossinline doSomething1: () -> Unit){Log.i("tag", "1")Runnable {doSomething1()}
}//调用
fun mainFun() {fun1 {Log.i("tag", "2")return@fun1//编译正常}Log.i("tag", "3")
}

结束。

http://www.cadmedia.cn/news/12530.html

相关文章:

  • html购物网站源代码新闻稿发布软文平台
  • 免费的查企业的网站seo搜索引擎招聘
  • 张家界网站seo如何优化标题关键词
  • 泰州做网站多少钱在线识别图片找原图
  • php主做哪种类型网站刷赞网站推广永久
  • b站视频未能成功转码深圳外贸推广公司
  • 连江网站建设服务seo优化便宜
  • 东莞网站建设公司注册杭州关键词优化外包
  • 软件工程师薪资赣州seo培训
  • 建设网站方向百度新闻排行榜
  • hbuilder 做网站网络服务提供商是指
  • 山西智能建站系统价格关键词首页排名代做
  • 华强北 做网站谷歌google浏览器
  • 深圳建设企业网站新闻20条摘抄大全
  • 网站站建设怎么在网上做广告
  • 龙岗网站建设需要考量些什么刷网站百度关键词软件
  • 泰安网站开发佛山seo教程
  • .案例 商务网站的推广策略设计网络营销方案
  • 中卫网站推广服务长沙seo推广
  • 宁波企业网站建设软文营销的写作技巧有哪些
  • 自助网站建设系统网站推广网站
  • 聊城市城乡建设委员会网站哈尔滨优化网站方法
  • 西宁招聘网站开发google ads 推广
  • 毕业论文做网站百度竞价专员
  • 全网网站建设链接交换平台
  • 百度公司官网seo网站优化推荐
  • 周口网站建设 网站制作 网络推广公司网站推广运营
  • 建设平台型网站多少钱如何宣传推广产品
  • 网站建设了解seo策略什么意思
  • wap 在线上海seo网站推广公司