有什么区别:

def even: Int => Boolean = _ % 2 == 0

and

val even: Int => Boolean = _ % 2 == 0

两者都可以称为偶数(10)。


当前回答

此外,Val是一个按值求值。这意味着右边的表达式在定义期间被求值。其中Def是通过名称求值。直到它被使用时才会计算。

其他回答

此外,Val是一个按值求值。这意味着右边的表达式在定义期间被求值。其中Def是通过名称求值。直到它被使用时才会计算。

在REPL,

scala> def even: Int => Boolean = { _% 2 == 0 }
even: Int => Boolean

scala> val even: Int => Boolean = { _% 2 == 0 }
even: Int => Boolean = $$Lambda$1157/1017502292@57a0aeb8

Def表示按名称调用,按需计算

Val表示在初始化时按值调用

考虑一下:

scala> def even: (Int => Boolean) = {
             println("def"); 
             (x => x % 2 == 0)
       }
even: Int => Boolean

scala> val even2: (Int => Boolean) = {
             println("val");
             (x => x % 2 == 0)
       }
val //gets printed while declaration. line-4
even2: Int => Boolean = <function1>

scala> even(1)
def
res9: Boolean = false

scala> even2(1)
res10: Boolean = false

你看到区别了吗?简而言之:

对于even的每一次调用,它都会再次调用even方法的主体。但是对于even2,即val,函数在声明时只初始化一次(因此它在第4行打印val,再也不打印了),并且每次访问都使用相同的输出。例如,试着这样做:

scala> import scala.util.Random
import scala.util.Random

scala> val x = { Random.nextInt }
x: Int = -1307706866

scala> x
res0: Int = -1307706866

scala> x
res1: Int = -1307706866

当x被初始化时,由Random返回的值。nextInt被设置为x的最终值。下一次再次使用x时,它总是返回相同的值。

你也可以惰性初始化x,即第一次使用它时,它是初始化的,而不是在声明时。例如:

scala> lazy val y = { Random.nextInt }
y: Int = <lazy>

scala> y
res4: Int = 323930673

scala> y
res5: Int = 323930673

执行像def x = e这样的定义不会求表达式e的值。相反,每当x被调用时,e都会被求值。

或者,Scala提供了一个值定义 Val x = e,它计算右边的值作为定义值的一部分。 如果随后使用x,则立即用预先计算的e值替换它,这样表达式就不需要再次求值。

瓦尔。“sq”在Scala定义中是固定的。它是在声明时计算的,以后不能更改。在其他例子中,even2也是val,但它声明了函数签名,即。"(Int => Boolean)",所以它不是Int类型。它是一个函数,它的值由下面的表达式设置

   {
         println("val");
         (x => x % 2 == 0)
   }

根据Scala的val属性,你不能将另一个函数赋值给even2,这与sq规则相同。

关于为什么调用eval2 val函数不打印“val”一次又一次?

源自代码:

val even2: (Int => Boolean) = {
             println("val");
             (x => x % 2 == 0)
       }

We know, in Scala last statement of above kind of expression (inside { .. }) is actually return to the left hand side. So you end up setting even2 to "x => x % 2 == 0" function, which matches with the type you declared for even2 val type i.e. (Int => Boolean), so compiler is happy. Now even2 only points to "(x => x % 2 == 0)" function (not any other statement before i.e. println("val") etc. Invoking event2 with different parameters will actually invoke "(x => x % 2 == 0)" code, as only that is saved with event2.

scala> even2(2)
res7: Boolean = true

scala> even2(3)
res8: Boolean = false

为了更清楚地说明这一点,下面是不同版本的代码。

scala> val even2: (Int => Boolean) = {
     |              println("val");
     |              (x => { 
     |               println("inside final fn")
     |               x % 2 == 0
     |             })
     |        }

会发生什么?在这里,当调用even2()时,我们看到“inside final fn”一遍又一遍地打印出来。

scala> even2(3)
inside final fn
res9: Boolean = false

scala> even2(2)
inside final fn
res10: Boolean = true

scala>