到本章这里,所有函数文本的例子仅参考了传入的参数。例如,(x: Int) => x > 0里,函数体用到的唯一变量,x > 0,是x,被定义为函数参数。然而也可以参考定义在其它地方的变量:

(x: Int) => x + more // more是多少?

函数把“more”加入参考,但什么是more呢?从这个函数的视点来看,more是个自由变量:free variable,因为函数文本自身没有给出其含义。相对的,x变量是一个绑定变量:bound variable,因为它在函数的上下文中有明确意义:被定义为函数的唯一参数,一个Int。如果你尝试独立使用这个函数文本,范围内没有任何more的定义,编译器会报错说:

scala> (x: Int) => x + more

<console>:5: error: not found: value more

(x: Int) => x + more

ˆ

另一方面,只要有一个叫做more的什么东西同样的函数文本将工作正常:

scala> var more = 1

more: Int = 1

scala> val addMore = (x: Int) => x + more

addMore: (Int) => Int = <function>

scala> addMore(10)

res19: Int = 11

依照这个函数文本在运行时创建的函数值(对象)被称为闭包:closure。名称源自于通过“捕获”自由变量的绑定对函数文本执行的“关闭”行动。不带自由变量的函数文本,如(x: Int) => x + 1,被称为封闭术语:closed term,这里术语:term指的是一小部分源代码。因此依照这个函数文本在运行时创建的函数值严格意义上来讲就不是闭包,因为(x: Int) => x + 1在编写的时候就已经封闭了。但任何带有自由变量的函数文本,如(x: Int) => x + more,都是开放术语:open term。因此,任何依照(x: Int) => x + more在运行期创建的函数值将必须捕获它的自由变量,more,的绑定。由于函数值是关闭这个开放术语(x: Int) => x + more的行动的最终产物,得到的函数值将包含一个指向捕获的more变量的参考,因此被称为闭包

这个例子带来一个问题:如果more在闭包创建之后被改变了会发生什么事?Scala里,答案是闭包看到了这个变化。如下:

scala> more = 9999

more: Int = 9999

scala> addMore(10)

res21: Int = 10009

直觉上,Scala的闭包捕获了变量本身,而不是变量指向的值。[5]就像前面演示的例子,依照(x: Int) => x + more创建的闭包看到了闭包之外做出的对more的变化。反过来也同样。闭包对捕获变量作出的改变在闭包之外也可见。下面是一个例子:

scala> val someNumbers = List(-11, -10, -5, 0, 5, 10)

someNumbers: List[Int] = List(-11, -10, -5, 0, 5, 10)

scala> var sum = 0

sum: Int = 0

scala> someNumbers.foreach(sum += _)

scala> sum

res23: Int = -11

例子用了一个循环的方式计算List的累加和。变量sum处于函数文本sum += _的外围,函数文本把数累加到sum上。尽管这是一个在运行期改变sum的闭包,作为结果的累加值,-11,仍然在闭包之外可见

如果闭包访问了某些在程序运行时有若干不同备份的变量会怎样?例如,如果闭包使用了某个函数的本地变量,并且函数被调用很多次会怎样?每一次访问使用的是变量的哪个实例?

仅有一个答案与语言余下的部分共存:使用的实例是那个在闭包被创建的时候活跃的。例如,以下是创建和返回“递增”闭包的函数:

def makeIncreaser(more: Int) = (x: Int) => x + more

每次函数被调用时都会创建一个新闭包。每个闭包都会访问闭包创建时活跃的more变量

scala> val inc1 = makeIncreaser(1)

inc1: (Int) => Int = <function>

scala> val inc9999 = makeIncreaser(9999)

inc9999: (Int) => Int = <function>

调用makeIncreaser(1)时,捕获值1当作more的绑定的闭包被创建并返回。相似地,调用makeIncreaser(9999),捕获值9999当作more的闭包被返回。当你把这些闭包应用到参数上(本例中,只有一个参数,x,必须被传入),回来的结果依赖于闭包被创建时more是如何定义的:

scala> inc1(10)

res24: Int = 11

scala> inc9999(10)

res25: Int = 10009

尽管本例中more是一个已经返回的方法调用的参数也没有区别。Scala编译器在这种情况下重新安排了它以使得捕获的参数继续存在于堆中,而不是堆栈中,因此可以保留在创建它的方法调用之外。这种重新安排的工作都是自动关照的,因此你不需要操心。请任意捕获你想要的变量:val,var,或参数。

最新文章

  1. CSS——关于z-index及层叠上下文(stacking context)
  2. A Complete List of .NET Open Source Developer Projects
  3. Toad for Oracle
  4. Android分享一款漂亮的折叠书架菜单
  5. iOS 根据字符串内容动态计算行高
  6. iOS中 视频直播功能-流媒体的使用(详解)韩俊强的CSDN博客
  7. break与continue
  8. REF CURSOR和CURSOR
  9. hdu1709(母函数)
  10. iOS开发之状态栏
  11. php中的数组遍历的几种方式
  12. linux下大于2T硬盘格式化方法
  13. POJ-2421 Constructing Roads---确定部分边的MST
  14. net core 随笔
  15. 关于anaconda中jupyter notebook错误
  16. React 体验
  17. (转)Android四大组件——Activity跳转动画、淡出淡入、滑出滑入、自定义退出进入
  18. BZOJ4445 SCOI2015小凸想跑步(半平面交)
  19. stm32 学习参考(转)
  20. ztree树形菜单的增加删除修改和换图标

热门文章

  1. HDU 1828 / POJ 1177 Picture (线段树扫描线,求矩阵并的周长,经典题)
  2. 10个免费开源的JS音乐播放器插件
  3. java 格式化代码 不进行换行
  4. php用fsockopen实现post提交数据并获得返回数据
  5. 唉,还是Windows好
  6. centos使用fuse挂载NTFS
  7. 使用RPM管理软件包
  8. ActionResult
  9. Spring boot 整合jsp和tiles模板
  10. 神经网络:卷积神经网络CNN