scala函数参数的传递:传值调用和传名调用

本文探讨了Scala中函数参数的两种传递方式——传值调用和传名调用。传值调用先计算参数值再应用,避免重复计算提高效率;传名调用则直接传递未计算的表达式,仅在需要时计算。通过实例分析,展示了两种调用方式的不同效果,帮助理解其工作原理。

函数参数的传递有两种:传值调用和传名调用
传值调用:先计算参数表达式的值,再应用到函数内部
传名调用:将未计算的参数表达式直接应用到函数内部

object Add{
def addByName(a: Int, b: =>Int) = a + b //传名调用
def addByValue(a: Int, b: Int)= a + b	//传值调用
}
addByName(2, 2 + 2)    
->2 + (2 + 2)
->2 + 4
->6
addByValue(2, 2 + 2)
->addByValue(2, 4)
->2 + 4
->6

传值调用与传名调用区别:

  • 传值调用避免了函数内部多次使用参数时重复计算其值,在一定程度上提高了效率
  • 传名调用的优势在于,如果参数在函数体内部没有被使用到,那么它就不用计算参数表达式的值

上边例子可能有些模糊,下面例子看懂了就会清晰很多。

object Test {var money = 10def drink: Unit ={money -= 1 }
def count: Int = {
					drink
					money }
def printByName(x: => Int): Unit = {for(i<-0 until 5)
println(x )
}
def printByValue(x: Int): Unit ={for(i<-0 until 5)
println( x )
}
def main(args:Array[String]) ={
printByName(count)
printByValue(count)
}}

代码实现后结果如下图:
在这里插入图片描述
前五次结果为传名调用的结果,每一次调用都会计算一次并且输出,而后五次的结果显然不同,传值调用直接算出了最终结果并且输出五次同样的结果。

下面还有一个例子,让我们更进一步了解他们的区别

import java.io.PrintWriter
object WhyAlwaysMe {
  var flag: Boolean = true
  def useOrNotUse(x: Int, y: =>Int) = {
    flag match{
      case true => x
      case false => x + y
    }
  }
  def main(args: Array[String]) =
  {
    println(useOrNotUse(1, 2))
    flag = false
    println(useOrNotUse(1, 2))
  }
}
本关任务:计算小明请完 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
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值