初识Continuation
本文来自网易云社区
作者:陆艺
上学时看了SICP之后就对scheme这个看上去比较古怪的语言产生了兴趣. 虽然书里并没有涉及scheme太多语法以及语言上特性的一些东西, 作为一个喜欢折腾的人, 手贱翻了翻Google, 了解到有continuation这样一个吊吊的东西, 于是借着有限的脑力学习了下.
基本概念
continuation的中文一般都叫做"续延"(还蛮好听的), 不过解释起来比较麻烦, 还是结合代码来看更好理解一点.
首先scheme里的continuation(太长了, 下文缩写为cont吧)的基本形式是长这个样子的:
(call/cc
(lambda (cc)
; do some computation)
)
call/cc
是call-with-current-continuation
的缩写, lambda的参数cc
就是当前环境的一个cont, 可以理解为当前的调用栈, 它知道自己未来需要执行的过程. 那么怎么表示"未来的计算"这样一个概念呢? 在编程语言中自然就是函数了, 而在scheme中就是过程(procedure), 更一般的就用lambda来表示. cc
就是这样一个只接受一个参数的lambda, 而调用(cc val)
则是整个cont动作的关键----它将流程跳回cont定义的位置, 并以val
代换为其计算出的结果.
来暴力理解一下, 把(call/cc ..) (rest code..)
看做这样一个结构[] (rest code ..)
, 在(cc val)
的时候直接回到了[]
的地方, 并将其替换为了val
.
看一个简单(似乎不是特别简单)的例子:
(+ 1 (call/cc)
(lambda (cc) (+ 2 (cc 3)))
)
一步步来:
整个block可以看做
(+ 1 [])
再看
[]
内部:(lambda (cc) (+ 2 (cc 3)))
, 这个lambda的body是会被执行的(cc 3)
以3为参数调用cont, 则跳转到其定义的地方[]
将其替换最后变为
(+ 1 3)
, 结果为4(+ 2 ..)
的部分并没有luan3用 :P
基本用法
Jump out
这个用法有个名字叫"非本地退出(non-local exit)", 说白了就是现代编程语言里break啦, exit啦这些东西.
scheme没有这些关键字
scheme觉得它们不够old-school
scheme决定这么搞
(define (search wanted? lst)
(call/cc
(lambda (return)
(for-each (lambda (e)
(if (wanted? e) (return e)))
lst)
#f
)
)
)
功能很直白, 从一个list里找到符合条件的元素, 不存在就返回#f
. return
的用法是直接从for-each
的循环中跳出了. 这里最后的#f
可不可以写成(return #f)
? 当然可以, 不过整个lambda都计算完了, 自然是返回最后一个计算值, 所以可以直接省略啦.
另一个例子
(define (list-product list)
(call/cc
(lambda (exit)
(let iter ((rest lst)) (cond
((null? rest) 1) ((zero? (car rest)) (exit 0)) (else (* (car rest) (iter (cdr rest))))
)
)
)
)
)
对一个list完成fold计算乘积的操作, 中途如果遇到0则立即退出, 避免了不必要的遍历.
Jump back
在scheme里, cont和lambda一样都是一等公民, 它自然也可以被当做参数抛来抛去, 或者跟其他变量进行绑定.
(define return #f)(+ 1 (call/cc
(lambda (cont) (set! return cont) 1)
)
)
老样子一步一步看:
全局定义一个
return
(绑定啥值无所谓)替换cont形式:
(+ 1 [])
在lambda body中,
return
与cont[]
绑定!1作为返回值, 则
(+ 1 [])
结果为2
然而并没有结束, 此时return
本身已经作为一个"+1器"存在了, 它代表的cont即(+ 1 [])
, 每次调用(return v)
便会得到v + 1
.
Jump out and Jump back
看到这里, 可以感觉到cont是能作为程序流控制的手段的, 用来完成一些比较时髦的动作, 比如大家都喜欢的协程 :P
所以接下来看一个模拟协程计算的较复杂的例子:
(define (hefty-computation do-other-stuff) ; 主要复杂计算
(let loop ((n 5))
(display "Hefty computation: ")(display n)(newline)
(set! do-other-stuff (call/cc do-other-stuff)) ; (1)
(display "Hefty computation (b)\n")
(set! do-other-stuff (call/cc do-other-stuff)) ; (3)
(display "Hefty computation (c)\n")
(set! do-other-stuff (call/cc do-other-stuff))
(if (> n 0) (loop (- n 1)))
)
) (define (superfluous-computation do-other-stuff) ; 次要计算
(let loop ()
(for-each (lambda (item)
(display item)(newline)
(set! do-other-stuff (call/cc do-other-stuff))) ; (2)
'("Straight up." "Quarter after." "Half past." "Quarter til.")
)
(loop) ; this trigger inf loop
)
) (hefty-computation superfluous-computation)
为了大家(我自己)能简单的看懂, 我们走的慢一些:
分别定义了两个过程: 主要计算和次要计算; 以调用主要计算开始, 参数为次要计算本身
开始主要计算, 进入loop(第一次), 打印"Hefty computation: 5"
(1)处
call/cc
以次要计算为参数, 结合后者定义, 则有(为了方便, 分别记两个两个过程的参数为do1和do2):主要计算产生第一个cont,
(set! do1 []-1)
开始次要计算, 参数do2为上一步的
[]-1
, 进入loop(第一次), for-each打印列表中第一个"Straight up."到达(2)
(call/cc do2)
, 此时次要计算产生一个cont, 记为[]-2
, 同时完成do2也就是[]-1
的计算, 流程回到了(1), 且将do1绑定到了[]-2
从(1)后继续主要计算, 打印"Hefty computation (b)"
主要计算(3)处产生第二个cont
[]-3
, 形式和上次一样; 由于参数do1实际上是[]-2
, 则:跳转到
[]-2
即(2)处继续执行, 此时set!
将[]-2
的返回值即[]-3
绑定到do2for-each的第二个循环打印"Quarter after."
流程又到(2)处,
(call/cc []-3)
, 并产生当前的cont[]-4
, 作为[]-3
的参数; 执行[]-3
的计算, 于是又回到了(3)处do1绑定为
[]-4
, 且打印"Hefty computation (c)"接下来的过程类似上面, 即不断的在主要计算和次要计算之间跳来跳去(和协程的切换是一样的)
看上去似乎有点晕, 总结一下这个用法的关键在于:
(call/cc cc)
实际上是将当前的cont作为参数contcc
的结果, 然后通过执行cc
跳转到cont创建时的地方; 之后通过set!
将之前的cont(即cc
的结果)保存下来, 然后再次通过(call/cc cont)
跳回去, 如此往复.
最后一个例子
(let* ((yin ((lambda (cc) (write-char #\@) cc) ; proc
(call/cc (lambda (c) c)))) ; arg
(yang ((lambda (cc) (write-char #\*) cc) ; proc
(call/cc (lambda (c) c))))) ; arg
(yin yang)
)
这是网上流传很久的yin-yang puzzle, 程序会不停的交替打印"@*@**@***@ ...". 这个例子大家可以尝试自己拿纸笔推一下(虽然过程特别的绕, 但是通过不断代换的笨办法还是可以一战的), 或者可以看一下这个解答就比较好理解了, stay calm :P
时至今日我仍然没有完整的学完scheme.. 这里也只是简单的介绍了下自己对continuation及用法的基本理解, 欢迎各位大大多加指正.
ref:
http://community.schemewiki.org/?call-with-current-continuation
http://stackoverflow.com/questions/2694679/how-does-the-yin-yang-puzzle-work/36513942#36513942
本文来自网易实践者社区,经作者陆艺权发布
相关文章:
【推荐】 深入解读Service Mesh背后的技术细节
【推荐】 网易云数据库架构设计实践
【推荐】 微服务监控探索
初识Continuation的更多相关文章
- kubernetes1.9管中窥豹-CRD概念、使用场景及实例
欢迎访问网易云社区,了解更多网易技术产品运营经验. 前言 默认读者有kubernetes基础概念的背景知识,因此基础概念例如有状态.pod.Replica Sets.Deployments.state ...
- Question | 移动端虚拟机注册等作弊行为的破解之道
本文来自网易云社区 "Question"为网易云易盾的问答栏目,将会解答和呈现安全领域大家常见的问题和困惑.如果你有什么疑惑,也欢迎通过邮件(zhangyong02@corp.ne ...
- appium封装显示等待Wait类和ExpectedCondition接口
此文已由作者夏鹏授权网易云社区发布. 欢迎访问网易云社区,了解更多网易技术产品运营经验. 使用WebDriver做Web自动化的时候,org.openqa.selenium.support.ui中提供 ...
- Android动画效果之初识Property Animation(属性动画)
前言: 前面两篇介绍了Android的Tween Animation(补间动画) Android动画效果之Tween Animation(补间动画).Frame Animation(逐帧动画)Andr ...
- 初识Hadoop
第一部分: 初识Hadoop 一. 谁说大象不能跳舞 业务数据越来越多,用关系型数据库来存储和处理数据越来越感觉吃力,一个查询或者一个导出,要执行很长 ...
- python学习笔记(基础四:模块初识、pyc和PyCodeObject是什么)
一.模块初识(一) 模块,也叫库.库有标准库第三方库. 注意事项:文件名不能和导入的模块名相同 1. sys模块 import sys print(sys.path) #打印环境变量 print(sy ...
- 初识IOS,Label控件的应用。
初识IOS,Label控件的应用. // // ViewController.m // Gua.test // // Created by 郭美男 on 16/5/31. // Copyright © ...
- UI篇(初识君面)
我们的APP要想吸引用户,就要把UI(脸蛋)搞漂亮一点.毕竟好的外貌是增进人际关系的第一步,我们程序员看到一个APP时,第一眼就是看这个软件的功能,不去关心界面是否漂亮,看到好的程序会说"我 ...
- Python导出Excel为Lua/Json/Xml实例教程(一):初识Python
Python导出Excel为Lua/Json/Xml实例教程(一):初识Python 相关链接: Python导出Excel为Lua/Json/Xml实例教程(一):初识Python Python导出 ...
随机推荐
- LG3690 【【模板】Link Cut Tree (动态树)】
题目 终于去写\(LCT\)了 这个大爷讲的挺好的 板子 #include<algorithm> #include<iostream> #include<cstring& ...
- 【[NOI2006]最大获利】
题目 并不知到为什么这道题讲了这么久 我们发现这道题就是最小割的板子啊,完全可以套上文理分科的板子 把每个机器和\(T\)连边,容量为\(p_i\),这些\(p_i\)并不计入总贡献 对于每一个要求我 ...
- SiteMesh配置下载使用(简单介绍)
简单介绍 SiteMesh 是一个网页布局和修饰的框架,利用它可以将网页的内容和页面结构分离,以达到页面结构共享的目的. Sitemesh是由一个基于Web页面布局.装饰以及与现存Web应用整合的框架 ...
- @WebListener 注解方式实现监听(eclipse和idea)
eclipse进行演示: 1.创建 Dynamic Web Project ,Dynamic Web module version选择3.0 2.在自动生成 的web.xml配置,增加 metadat ...
- Android学习笔记_67_Android MyCrashHandler 中异常处理 UncaughtExceptionHandler
1.程序中故意抛出异常: public class ExceptionActivity extends Activity { String str; @Override public void onC ...
- JavaScript函数的方法
在一个对象中绑定函数,称为这个对象的方法. 在JavaScript中,对象的定义是: var xiaoming = { name:'小明'; birth:1990; }; 但是,如果我们给xiaomi ...
- 菜鸟笔记 -- Chapter 6 面向对象
在Java语言中经常被提到的两个词汇是类与对象,实质上可以将类看作是对象的载体,它定义了对象所具有的功能.学习Java语言必须要掌握类与对象,这样可以从深层次去理解Java这种面向对象语言的开发理念, ...
- 2018 Wannafly summer camp Day3--Knight
Knight 题目描述: 有一张无限大的棋盘,你要将马从\((0,0)\)移到\((n,m)\). 每一步中,如果马在\((x,y)(x,y)\),你可以将它移动到 \((x+1,y+2)(x+1,y ...
- Linux外在设备的使用
参考高峻峰 著 循序渐进Linux(第二版) 软盘在linux下对应的设备文件为/dev/fdx.主设备号fd是软盘驱动器(floppydisk)的缩写,次设备号x是软盘驱动器的相应编号.例如,/de ...
- (第02节)集成Sping框架
通过第一节创建好的Web项目,接下来就是集成Spring框架 首先让我们看下创建好的Web项目的基本结构 其中,Java跟test是我自己创的,然后就是一般的webapp文件,和pom配置文件,要在w ...