编译此代码段时,scala编译器会发出以下警告:
不推荐使用零参数方法值的Eta扩展。你是否 打算写Main.this.porFiles5()? [warn] timerFunc(porFiles5)
当我将函数传递给另一个函数以进行简单的计时时,会发生这种情况。 timer函数采用无参数函数返回单元,在此行:timerFunc(porFiles5)
。这个警告是否必要?什么是避免它的惯用方法?
package example
import java.nio.file._
import scala.collection.JavaConverters._
import java.time._
import scala.collection.immutable._
object Main extends App {
val dir = FileSystems.getDefault.getPath("C:\\tmp\\testExtract")
def timerFunc (func:()=>Unit ) = {
val start = System.currentTimeMillis()
timeNow()
func()
val finish = System.currentTimeMillis()
timeNow()
println((finish - start) / 1000.0 + " secs.")
println("==================")
}
def porFiles5(): Unit = {
val porFiles5 = Files.walk(dir).count()
println(s"You have $porFiles5 por5 files.")
}
def timeNow(): Unit = {
println(LocalTime.now)
}
timeNow()
timerFunc(porFiles5)
timeNow()
}
答案 0 :(得分:12)
porFiles5
不是一个函数。它是方法,在Scala中完全不同。
如果你有一个方法,但是你需要一个函数,你可以使用η-expansion将方法提升到一个函数中,如下所示:
someList.foreach(println _)
在某些情况下,Scala会自动执行η-expansion,如果从上下文中绝对清楚你的意思,例如:
someList.foreach(println)
然而,无参数方法存在歧义,因为Scala允许您在没有参数列表的情况下调用无参数方法,即可以在没有任何参数列表的情况下调用使用 empty 参数列表定义的方法。所有:
def foo() = ???
foo // normally, you would have to say foo()
现在,在您的情况下,存在歧义:您是指调用 porFiles5
还是意味着η-扩展它?目前,Scala任意消除这种情况的歧义并执行η-expansion,但在将来的版本中,这将是一个错误,您将不得不明确执行η-expansion。
因此,要摆脱警告,只需使用显式η-expansion而不是隐式η-expansion:
timerFunc(porFiles5 _)