一个例子让你明白Scala中的call by name和call by value

 

先看一个例子

def get(x:Int):Int={
println(s"$x")
x
}

def getByValue(y:Int):Int={
x+x
}


def getByName(y: =>Int):Int={
x+x
}

getByValue(get(1))
getByName(get(2))

 (1)get方法

该方法传入x,先打印x,然后将x作为返回值。

(2)getByValue和getByName

这两个方法定义非常相似,参数写法不同,getByName参数类型是=>Int。而GetByValue的参数类型是Int。

它们有什么区别呢?

getByValue(get(1))中通过get(1)方法将值传递给了y,即y=get(1),此时调用get(1)方法,所以输出1,

getByName(get(2))中通过get(2)方法将get(1)的地址给了y,每次需要x都要根据这个地址取找具体的值,getByName方法中有两次使用x,每一次都要根据地址来找x,所以输出两次2.

输出结果:

1
2
2

 call by value和call by name的不同之处,可以如下示意理解:

call by value在用到y的时候,直接把x的值给了y,之后它俩再无关系;

call by name在用到y的时候,是把x的地址给了y,值并没有给他,每次用y的时候,先由地址找到x的值。

很类似C语言的函数参数的值传递和地址传递,不过上面的示意图只是帮助大家理解。

举个生活中的例子,比如做包子需要面粉,call by value是把面粉直接买回来,做包子的时候直接用就可以,而call by name是把面粉店的地址记下来,每次做包子的时候,先根据面粉店的地址找到面粉店,然后从面粉店把面粉买回来,才做包子。 

 

本关任务:计算小明请完 5 顿饭之后还剩多少钱。 相关知识 为了完成本关任务,你需要掌握: 什么是函数; 如何调用函数。 函数定义 Scala 作为支持函数式编程的语言,Scala 函数式编程是 Scala 的重中之重,高级函数也是其独特的一个特性。首先,函数/变量同是一等公民,函数与变量同等地位,函数的定义可以单独定义,可以不依赖于类、接口或者 object,可以独立存在、独立使用,并且可以赋值给变量。 我们上一关所学习的方法就是函数的一种形式。接下来我们来看函数的定义: (函数的参数列表)=>{函数体内容} 由于该函数需要被别人调用,所以要用一个变量来引用。 val 变量名称 =(函数的参数列表)=>{函数体内容} 示例: // 将 x 的值加 1 val increase = (x:Int) => x+1 => 表示该函数将左侧的内容(任何整数 x)转换成右侧的内容(x+1)。 因此,这是一个将任何整数 x 映射成 x+1 的函数。 复杂的定义形式: val 变量名称:(输入参数类型) => 返回值类型 =(输入参数的引用)=>{函数体内容} 示例: // 计算 x y 的 val add: (Int, Int) => Int = (x:Int, y:Int)=> x+y 如果你想要在函数中包含多于 1 条语句,可以将函数体用花括号括起来,每条语句占一行,组成一个代码块(block)。跟方法一样,当函数值被调用时,所有的语句都会被执行,并且该函数的返回值就是对最后一个表达式求值的结果。 参数列表对于方法是可选的,但是对于函数是强制的。 方法可以没有参数列表,参数列表也可以为空。但是函数必须有参数列表(参数列表可以为空)。 而函数可以直接赋值给变量,可以让函数很方便的传递。 函数调用 传值调用传名调用 Scala 的解释器在解析函数参数(function arguments)时有两种方式: 传值调用(call-by-value):先计算参数表达式的值(reduce the arguments),再应用到函数内部; 传名调用(call-by-name):将未计算的参数表达式直接应用到函数内部。 object Test { def main(args: Array[String]): Unit = { callByValue(something()) println("------------------------------") callByName(something()) } def something() = { println("calling something") } def callByValue(x: Int) = { println("x1=" + x) println("x2=" + x) } def callByName(x: => Int) = { println("x1=" + x) println("x2=" + x) } } 执行结果: calling something x1=1 x2=1 ------------------------------ calling something x1=1 calling something x2=1 从执行结果上来看,我们可以得出: 在进入函数内部前,传值调用方式就已经将参数表达式的值计算完毕,而传名调用是在函数内部进行参数表达式的值计算的。 这就造成了一种现象,每次使用传名调用时,解释器都会计算一次表达式的值。 指定函数参数名调用 一般情况下函数调用参数,就按照函数定义时的参数顺序一个个传递。但是我们也可以通过指定函数参数名,并且不需要按照顺序向函数传递参数。 object Test { def main(args: Array[String]) { printInt(7,5); // 默认按参数顺序调用 println("------------------------------") printInt(b=5, a=7); // 指定函数参数名调用 } def printInt( a:Int, b:Int ) = { println("Value of a : " + a ); println("Value of b : " + b ); } } 执行结果: Value of a : 7 Value of b : 5 --------------- Value of a : 7 Value of b : 5 注意:当参数列表为空时,直接写方法名可以调用方法,而函数名只是代表函数对象本身。因为保存函数字面量的变量(又称为函数名或者函数值)本身就是实现了 FunctionN 特质的类的对象
最新发布
04-03
### 计算剩余金额逻辑的 Scala 实现 在 Scala 中,函数是一等公民[^1],因此可以灵活地定义调用函数来解决问题。下面是一个关于计算剩余金额的具体实现。 #### 定义函数 可以通过多种方式定义函数[^2]。对于计算剩余金额的问题,可以选择标准形式定义函数: ```scala def calculateRemainingAmount(total: Double, spent: Double): Double = { total - spent } ``` 上述代码中,`calculateRemainingAmount` 是一个接受两个 `Double` 类型参数的函数,并返回两者的差值作为剩余金额的结果。 #### 调用函数 调用该函数时可以直接传入参数或者通过指定参数名称的方式进行调用[^4]。例如: ```scala val totalAmount = 100.0 val amountSpent = 35.0 // 按照顺序传递参数 val remainingAmount1 = calculateRemainingAmount(totalAmount, amountSpent) // 使用命名参数 val remainingAmount2 = calculateRemainingAmount(spent = amountSpent, total = totalAmount) println(s"剩余金额 (按序传递) : $remainingAmount1") // 输出 剩余金额 (按序传递) : 65.0 println(s"剩余金额 (命名参数) : $remainingAmount2") // 输出 剩余金额 (命名参数) : 65.0 ``` #### 将函数赋值给变量 由于函数是一等公民[^3],还可以将函数赋值给变量以便于后续多次使用: ```scala val calcRemain = calculateRemainingAmount _ val result1 = calcRemain(200.0, 80.0) val result2 = calcRemain(150.0, 50.0) println(s"结果1 : $result1") // 输出 结果1 : 120.0 println(s"结果2 : $result2") // 输出 结果2 : 100.0 ``` 以上展示了如何利用 Scala 的特性完成简单的业务逻辑——计算剩余金额。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ErbaoLiu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值