18

作为练习,我尝试创建一个隐式转换,它可以接受一个函数并生成一个Runnable. 这样,您可以调用接受Runnable对象的 Java 方法并像闭包一样使用它们。

隐式转换很容易:

    implicit def funToRunnable(fun : Unit) = new Runnable() { def run = fun }

但是我不知道怎么称呼它。如何传入一个返回 Unit 的无参数函数,而不立即对其进行评估?例如,我希望以下内容打印“12”,但它打印“21”,因为print("2")它是一次评估的。

    var savedFun : Runnable = null
    def save(r : Runnable) = { savedFun = r }

    save(print("2"))
    print("1")
    savedFun.run()

我如何告诉编译器将其print("2")视为函数的主体,而不是立即评估的东西?我尝试了一些可能性,例如

    save(() => print("2"))

或者

    save(=> print("2"))

不是合法的语法。

4

5 回答 5

24

arg,刚刚回答了我自己的问题。我错误地实现了隐式转换。正确的实现是

implicit def funToRunnable(fun: () => Unit) = new Runnable() { def run() = fun() }

你这样称呼它:

save(() => print("2"))

这将产生“2”

于 2010-06-18T23:40:25.723 回答
14

如果你想危险地生活,你可以将任何东西转换为可运行的:

implicit def whateverToRunnable[F](f: => F) = new Runnable() { def run() { f } }

scala> val t = new Thread(println("Hello"))
t: java.lang.Thread = Thread[Thread-2,5,main]

scala> t.start()
Hello

或者您可以创建自己的线程创建者和启动器:

def thread[F](f: => F) = (new Thread( new Runnable() { def run() { f } } )).start

scala> thread { println("Hi"); Thread.sleep(1000); println("Still here!") }
Hi

scala> Still here!

如果你想返回线程,那么

def thread[F](f: => F) = {
  val t = new Thread( new Runnable() { def run() { f } } )
  t.start()
  t
}

但是所有这些,虽然有用,但可能甚至不如scala.actors.Futures(仅在 2.8 上测试过)有用:

scala> import scala.actors.Futures

scala> val x = Futures.future { Thread.sleep(10000); "Done!" }
x: scala.actors.Future[java.lang.String] = <function0>

scala> x.isSet
res0: Boolean = false

scala> x.isSet
res1: Boolean = false

scala> x()   // Waits until the result is ready....
res2: java.lang.String = Done!
于 2010-06-19T02:09:49.397 回答
5

实际上,您可以使用按名称调用参数做得更好:

implicit def runnable(f: => Unit): Runnable = new Runnable() { def run() = f }

用法:

import concurrent.ExecutionContext.Implicits.global._
execute(print("hello"))
于 2015-04-20T21:30:55.467 回答
5

有趣的是,通过这种方式,您可以执行接收 aRunnable并将其传递给闭包的代码。

看:

scala> new Thread( ()  => print( "Hello" ) ).start()
<console>:5: error: overloaded method constructor Thread with alternatives (java.lang.ThreadGroup,java.lang.Runnable,java.lang.String,Long)java.lang.Thread <and> (java.lang.ThreadGroup,java.lang.Runnable,java.lang.String)java.lang.Thread <and> (java.lang.Runnable,java.lang.String)java.lang.Thread <and> (java.lang.ThreadGroup,java.lang.String)java.lang.Thread <and> (java.lang.String)ja...
       new Thread( ()  => print( "Hello" ) ).start()


scala> implicit def funcToRunnable( func : () => Unit ) = new Runnable(){ def run() = func() }
funcToRunnable: (() => Unit)java.lang.Object with java.lang.Runnable

scala> def doRun( runnable: Runnable ) = runnable.run
doRun: (Runnable)Unit

scala> doRun( () => print("Hola"))
Hola

scala> new Thread(()=>print("Hello")).start()

scala> Hello
于 2010-06-18T23:54:56.533 回答
0

在另一个线程中运行一些代码的另一种方法:

scala.actors.Actor.actor { ...doSomething()... }
于 2012-05-03T11:54:44.170 回答