1

在我学习 Scala 的过程中,我尝试用回调实现一个简单的 DSL

object Button {...} // apply 
class Button(val name: String) {
    private val: => Unit; // doesn't work

    def click(f: => Unit) = {
        _click_cb = f
        this
    }

    def onClick() = this._click_cb()
}

Button("Click me!") click {println("Clicked!")}

我创建了一个新对象,将它传递给存储的回调。我的演示框架触发onClick方法,应该调用存储的方法

它适用,() => Unit但我的 DSL 看起来很丑:

Button("Click me!") click (() => println("Clicked!"))

当然,我可以onClick稍后做抽象并实现一个匿名类

new Button("Click me!") {def onClick = println("Clicked!")}

但我想玩一些 DSL 之类的

问题是:

  • f我该如何存放_click_cb
  • 如何提供初始“空”功能_click_cb
  • 也许有更多的scala方式来实现这一点?(没有匿名类)
4

1 回答 1

2

一个丑陋的版本只是为了表明惰性 val 可以保存按名称参数值而不对其进行评估:

case class Button(val name: String) {
  def clickCallback(): Unit = ()

  def click(f: => Unit) = {
    lazy val notEvaluated = f
    new Button(name) { override def clickCallback() = notEvaluated }
  }

  def onClick(): Unit = clickCallback()
}

更清洁、更实用的实现:

class Button(val name: String) {
  def click(f: => Unit) = new Button(name) { override def onClick() = f }

  def onClick(): Unit = ()
}
于 2015-05-14T07:24:27.443 回答