[软件设计/软件工程] DSL 的 Kotlin sink 函数需要说明

[复制链接]
发表于 2022-5-3 11:22:42
问题
我正在尝试理解以下代码
  1. class HTML {
  2.     fun body() { ... }
  3. }

  4. fun html(init: HTML.() -> Unit): HTML {
  5.     val html = HTML()  // create the receiver object
  6.     html.init()        // pass the receiver object to the lambda
  7.     return html
  8. }

  9. html {       // lambda with receiver begins here
  10.     body()   // calling a method on the receiver object
  11. }
复制代码

我真正无法理解的是

html.init() // 将接收者对象传递给 lambda

这里发生了什么?

有人可以简要解释一下这里发生了什么吗?

回答
首先,让这个例子保持简单,看看问题出在哪里。

我们可以像这样构建 html 函数:
  1. fun html(init: (HTML) -> Unit): HTML {
  2.     val html = HTML()
  3.     init(html)
  4.     return html
  5. }
复制代码

这将更容易理解(起初),因为我们只是将通常的单参数 lambda 传递给 html 函数。

但是现在调用站点不像构建器:
  1. html { it: HTML -> // just for clarity     
  2.     it.body() // not very nice
  3. }
复制代码

如果我们可以在没有 body() 的情况下在 html 中调用它,那不是很好吗?这是可能的!我们只需要一个带有接收器的 lambda。
  1. fun html(init: HTML.() -> Unit): HTML { // <-- only this changed
  2.     val html = HTML()
  3.     init(html)
  4.     return html
  5. }
复制代码

请参阅如何像以前一样将 html 作为参数传递给 init?

当然,我们也可以这样调用它:html.init(),如示例所示。 HTML 的实例在 lambda 块内变成了 this。

现在,我们可以这样做:
  1. html {      
  2.    this.body()
  3. }
复制代码

由于可以省略,因此我们在这里:
  1. html {      
  2.    body()
  3. }
复制代码

所以最终 lambdas 和接收器使代码更简洁,并允许我们使用良好的生成器语法。





上一篇:是否以编程方式访问 Android Q (SDK=29) 上的下载文件夹?
下一篇:MongoDB未找到数据:返回未定义

使用道具 举报

Archiver|手机版|小黑屋|吾爱开源 |网站地图

Copyright 2011 - 2012 Lnqq.NET.All Rights Reserved( ICP备案粤ICP备14042591号-1粤ICP14042591号 )

关于本站 - 版权申明 - 侵删联系 - Ln Studio! - 广告联系

本站资源来自互联网,仅供用户测试使用,相关版权归原作者所有

快速回复 返回顶部 返回列表