A function object that captures free variables, and is said to be
“closed” over the variables visible at the time it is created is called closure. For example:
var foo = 10
val bar = (x:Int) => x+foo
Here, foo
is free variable in function literal and hence the function literal is open term. As a result, val bar
is a closure since function value (object) is created with open term foo
.
Now, lets start with different between val and def in term of function definition.
In scala, function is a value and you can assign it to variable.
When you define a function with val, you are defining it with a function literal as val bar = (x:Int) => x+foo
, which return a function object and assign it as value to bar
. In case of val, the function value is evaluated as it is defined. That means, function literal (x:Int) => x+foo
is evaluated to function value with free variable foo
and is stored in bar
variable as closure.
scala> var foo = 10
foo: Int = 10
scala> val bar = (x: Int) => x + foo
bar: Int => Int = <function1>
Since, bar
is a value and is evaluated as it is defined, it will always refer to same object in the memory whenever accessed.
scala> bar eq bar
res11: Boolean = true
On the other hand, you define a method with def and method is not a function in Scala. According to Scala language specification, method do not have type in Scala hence it cannot be used as value.
For example, you cannot do as:
val bar = {def foo(x:Int): Int = x + 1 }
However, according to Specification, if method name is used as value, then Scala implicitly converts it to corresponding function type with ETA Expression. e.g. method def a: Int
is converted as a: => Int
. That means, every time you call method, it returns a function value.
scala> def foo():Int = 1
a: ()Int
scala> val bar:( ()=> Int) = foo
bar: () => Int = <function0>
Because of this, method can be used as function. For example, when ever some method or function require a function type as argument, you can provide def method.
scala> val foo = (x:Int, double:Function1[Int, Int]) => double(x)
foo: (Int, Int => Int) => Int = <function2>
scala> def double(x:Int):Int = x * 2
double: (x: Int)Int
scala> foo(3, double)
res1: Int = 6
In addition, note that with method, you get a new function on every call.
scala> def double: Int => Int = _ * 2
double: Int => Int
scala> double eq double
res15: Boolean = false
Now, lets go to closure.
Both function literal defined with val and def method return function value (object). The function value (the object) that’s created at runtime from function literal is closure. On the other hand, a method is not a closure, but the function value you get from calling a method is a closure.
scala> var more = 1
more: Int = 1
scala> val foo = (x:Int) => x + more
foo: Int => Int = <function1>
scala> def bar(x:Int):Int = x + more
bar: (x: Int)Int
scala> val barClouser : (Int => Int) = bar
barClouser: Int => Int = <function1>
scala> val barValue = bar(3)
barValue: Int = 4
scala> val fooValue = foo(3)
fooValue: Int = 4
val
and one defined usingdef
", the closure characteristic has nothing to do there – cchantep