Call By Value vs. Call By Name Evaluation
data:image/s3,"s3://crabby-images/d9197/d9197277107c9fd4a754977ab8f9d1d28c8fbd5f" alt="Animesh Gupta"
data:image/s3,"s3://crabby-images/ec3b3/ec3b368127270e803111fedc0cc461e9f6ade303" alt=""
Mostly all programming languages follow two primary ways of evaluating function calls.
Call by Value / Eager evaluation
Call by Name / Lazy Evaluation
Both evaluating strategies reduce to the same final value as long as the following hold true ::
The reduced expression consists of pure functions.
Both evaluations terminate.
Que. How does the compiler and further the execution environment identifies whether a function or method is call by value or call by name ?
Ans. The compile time syntax identifies call by value or call by name mechanism. The execution environment follows the function definition to decide between call by value and call by name.
Some languages optimize the “call by name” mechanism into “call by need”, i.e., lazy evaluation only by default to avoid repeated computations. e.g. Haskell
Que. When to use Call by Value evaluation strategy ?
Ans. Call by value finds it’s usefulness when we want efficiency and predictability of execution. Also, it has the advantage of evaluating every function argument only once. e.g. enabling a counter, concatenating outputs incrementally, etc.
Que. When to use Call by Name evaluation strategy ?
Ans. Call by name shines when a function argument is either not at all used in the function body or we need to evaluate the argument again and again from scratch. This strategy is best for lazy evaluation, performance optimization and for avoiding unnecessary evaluations. e.g. mapping rows from a SQL query into objects where the task of mapping for each row needs to be from beginning to end, again and again.
Language Specific Design
Language | Default Strategy | Supports Call Name ? |
C | Call by Value | N/A |
C++ | Call by Value | N/A |
Java | Call by Value | Simulated using Supplier<T> |
Scala | Call by Value | Yes |
Python | Call by Value | Simulated using Lambdas |
Lisp | Call by Value | Yes |
Julia | Call by Value | Simulated using Thunks |
Haskell | Call by Need | Yes |
Go | Call by Value | Simulated using ( func() ) |
Rust | Call by Value | Simulated using Closures |
JavaScript | Call by Value | Simulated using Higher Order Functions |
Call By Value Example with Output using Scala coding language.
Note :: Remember, in Call by Value evaluation strategy, arguments are evaluated before passing to the function.
object SumOfSquaresByValue {
def sumOfSquares(x: Int, y: Int): Int = {
x * x + y * y
}
def main(args: Array[String]): Unit = {
val a = 3
val b = 4
println(s"Sum of squares of $a and $b: " + sumOfSquares(a, b))
}
}
Output ::
Sum of squares of 3 and 4: 25
Call By Name Example with Output using Scala coding language.
Note :: Remember, in Call by Name evaluation strategy, arguments are evaluated each time they are accessed inside the function.
object SumOfSquaresByName {
def sumOfSquares(x: => Int, y: => Int): Int = {
println(s"Evaluating x: $x")
println(s"Evaluating y: $y")
x * x + y * y
}
def main(args: Array[String]): Unit = {
var a = 3
var b = 4
def computeA(): Int = {
println("Computing a...")
a
}
def computeB(): Int = {
println("Computing b...")
b
}
println("Calling sumOfSquares...")
println("Result: " + sumOfSquares(computeA(), computeB()))
}
}
Output ::
Calling sumOfSquares...
Computing a...
Evaluating x: 3
Computing b...
Evaluating y: 4
Computing a...
Computing b...
Result: 25
Approach | Evaluation Strategy | Reusability | Behaviour |
Call by Value | Evaluates before function call | Simple | Efficient when value used multiple times |
Call by Name | Evaluates each time it is accessed | Useful for lazy computations | May cause repeated evaluations |
Subscribe to my newsletter
Read articles from Animesh Gupta directly inside your inbox. Subscribe to the newsletter, and don't miss out.
Written by
data:image/s3,"s3://crabby-images/d9197/d9197277107c9fd4a754977ab8f9d1d28c8fbd5f" alt="Animesh Gupta"