前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Scala基础 - 传名参数和传值参数(by-name/by-value parameter)

Scala基础 - 传名参数和传值参数(by-name/by-value parameter)

作者头像
joymufeng
发布2018-05-17 15:44:01
1.8K0
发布2018-05-17 15:44:01
举报

1. 介绍

传值参数在函数调用之前表达式会被求值,例如Int,Long等数值参数类型;传名参数在函数调用前表达式不会被求值,而是会被包裹成一个匿名函数作为函数参数传递下去,例如参数类型为无参函数的参数就是传名参数。下文将举例说明这两者的区别。

2. 简单类型的传递示例

2.1 传值参数(by-value parameter)示例

在下面的示例中,编译器检测到strToInt接受一个传值参数,所以先对传入的参数表达式{println("eval parameter expression"); "123"},然后再讲求值结果传递给strToInt。

代码语言:javascript
复制
object Test {
  def strToInt(s: String) = {
    println("call strToInt")
    s.toInt
  }

  def main(args: Array[String]) {
    strToInt({println("eval parameter expression"); "123"})
  }
}
//输出:
eval parameter expression
call strToInt
2.2. 传名参数(by-name parameter)示例

strToInt函数声明中添加一个=>,参数s的类型就变成了无参函数,类型为:() => String,按照Scala针对无参函数的简化规则,此处省略了()。因为参数s的类型是无参函数,所以此处是按名传递。

代码语言:javascript
复制
object Test {
  def strToInt(s: => String) = {
    println("call strToInt")
    s.toInt
  }

  def main(args: Array[String]) {
    strToInt({println("eval parameter expression"); "123"})
  }
}
//输出:
call strToInt
eval parameter expression

从上面的输出可以看出,参数表达式在strToInt函数调用之后才被求值。其实此处编译器自动将参数表达式{println("eval parameter expression"); "123"}转换成匿名的无参函数,并传递给s

3. 复杂类型的传递示例

3.1 传值参数(by-value parameter)示例

invode函数的参数f的类型为柯里化函数String => Int => Long, 此处为按值传递。

代码语言:javascript
复制
object Test {
  def invode(f: String => Int => Long) = {
    println("call invoke")
    f("1")(2)
  }

  def curry(s: String)(i: Int): Long = {
    s.toLong + i.toLong
  }

  def main(args: Array[String]) {
    invode{println("eval parameter expression");curry}
  }
}
//输出:
eval parameter expression
call invoke
3.2 传名参数(by-name parameter)示例

invode函数的参数f的类型为一个无参函数,该无参函数的返回类型为柯里化函数String => Int => Long, 因为参数f的类型是无参函数,所以此处是按名传递。

代码语言:javascript
复制
object Test {
  def invode(f: => String => Int => Long) = {
    println("call invoke")
    f("1")(2)
  }

  def curry(s: String)(i: Int): Long = {
    s.toLong + i.toLong
  }

  def main(args: Array[String]) {
    invode{println("eval parameter expression");curry}
  }
}
//输出:
call invoke
eval parameter expression

4. 小结

如果参数类型是无参函数,则按名传递,否则按值传递。注意,如果参数类型是函数类型,但不是无参函数,还是按值传递,例如:

代码语言:javascript
复制
object Test {
  def strToInt(s: (String) => Int) = {
    println("call strToInt")
    s("123")
  }

  def main(args: Array[String]) {
    strToInt({println("eval parameter expression"); (s: String) => s.toInt})
  }
}
//输出:
eval parameter expression
call strToInt
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1. 介绍
  • 2. 简单类型的传递示例
    • 2.1 传值参数(by-value parameter)示例
      • 2.2. 传名参数(by-name parameter)示例
      • 3. 复杂类型的传递示例
        • 3.1 传值参数(by-value parameter)示例
          • 3.2 传名参数(by-name parameter)示例
          • 4. 小结
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档