控制结构(11): Continuation passing style(CPS)
// 上一篇:控制结构(10)指令序列(opcode)
[注释]:
- 这个笔记系列需要告一个段落了,收尾部分整理下几个时髦(The New Old Things)结构。
- 后面打算开一个算法方面的,重新学习下算法方面的知识。
前面的文章里,在所有异步的地方都只假设通过callback的方式实现,而不用其他新的原子结构。实际上,如果我们进一步要求:
- 不使用return函数返回
- 一个函数执行的最后一步要么退出,要么调用其他函数来传递结果。
那么,这样的一个函数在最后一个动作之后是不需要保存Stack的。如果这个接收结果的被调函数本身总是以函数参数的形式传递,则我们称这样的手法为“CPS”:
In functional programming, continuation-passing style (CPS) is a style of programming in which control is passed explicitly in the form of a continuation. This is contrasted with direct style, which is the usual style of programming. --wiki:CPS
wiki里面举例以lisp函数语言的case居多,对于习惯命令式语言的人来说,阅读起来可能感觉不是很强烈。以JavaScript的视角,可以阅读下面这三篇文章:
- https://blogs.msdn.microsoft.com/ericlippert/2005/08/08/recursion-part-four-continuation-passing-style/
- https://blogs.msdn.microsoft.com/ericlippert/2005/08/11/recursion-part-five-more-on-cps/
- https://blogs.msdn.microsoft.com/ericlippert/2005/08/15/recursion-part-six-making-cps-work/
以下翻译上述链接里面的例子.
假设有这样一个命令式的递归函数:
function calc(l,r){
return l+r;
}
function vist(tree){
if(tree==null){
return 0;
}else{
var l = vist(tree.left);
var r = vist(tree.right);
return calc(l,r);
}
}
我们希望一步步把它改写成CPS风格的代码,第一步把visit函数的返回去掉,改用一个回调函数k来接收结果:
function vist(tree, k){
if(tree==null){
k(0);
}else{
// var l = vist(tree.left);
// var r = visit(tree.right);
k(calc(l,r));
}
}
注意到上面的代码注释了两行,因为我们希望visit没有返回值,则这两行代码不能再调用visit的原版代码。怎么做呢?
通过反复执行下面的步骤来达成:
- 被注释掉的部分,最后一个操作的结果应该传递给一个“回调函数”,假设是C
- 使用了该结果的后续代码应该被“卷入”到C里面。
于是上述代码变为:
function visit(tree,k){
if(tree==null){
k(0);
}else{
// var l = vist(tree.left);
// 这个callback消灭了var r = visit(tree.right);和k(calc(l,r));
function kRight(r){
k(calc(l,r));
}
// 此时可以把kRight作为visit的第2个参数,这是一个CPS
visit(tree.right, kRight);
}
}
如此,我们还差一个// var l = vist(tree.left);没有消灭,再来一次:
function visit(tree, k){
if(tree==null){
k(0);
}else{
function kLeft(l){
function kRight(r){
k(calc(l,r));
}
visit(tree.right, kRight);
}
visit(tree.left, kLeft);
}
}
除了calc那个地方依然是命令式代码,其他部分都已经是CPS风格代码。但这个地方的消除就留给读者做一个练习。当一个代码全部都变成函数调用的时候,我们可以让每个函数调用的地方实际上都变成生成包含发生这个函数调用所需要的数据,而不是直接执行它。但在此之前,我们先让上面函数的参数合并成一个(一个简单的结构体就可以搞定它)。
function visit(args){
if(args.tree==null){
args.k(0);
}else{
function kLeft(l){
function kRight(r){
args.k(calc(l,r));
}
visit(args.tree.right, kRight);
}
visit(args.tree.left, kLeft);
}
}
好了,现在,我们先造一个生成包含发生这个函数调用所需要的数据的辅助函数:
var continuation = null;
function createContinuation(newFun,newArgs){
continuation = {func: newFun, args: newArgs};
}
那么,visit函数就会被改写成一个让每个函数调用的地方实际上都变成生成包含发生这个函数调用所需要的数据,而不是直接执行它的函数:
function visit(args){
if(args.tree==null){
createContinuation(k, 0);
}else{
function kLeft(l){
function kRight(r){
//args.k(calc(l,r));
createContinuation(k, calc(l,r));
}
//visit(args.tree.right, kRight);
createContinuation(visit, {tree:args.tree.right,k:kRight});
}
//visit(args.tree.left, kLeft);
createContinuation(visit, {tree:args.tree.left,k:kLeft});
}
}
那么,原来的调用哪去了呢?现在这个visit似乎只是不断在改写全局变量continuation而已?别急,马上你就会明白怎么回事了:
function runIterate(){
while(continuation!=null){
var f = continuation.func;
var args = continuation.args;
continuation = null;
f(args); //思考:此处发生了什么?
}
}
runIterate函数可以轻易的在 continuation被改写的间隙执行这个 continuation所代表的函数动作!调用如下:
createContinuation(visit, {tree: mytree, k: print});
runIterate();
当然,可以改写为递归的执行:
function runRecursive(){
if(continuation){
var f = continuation.func;
var args = continuation.args;
continuation = null;
f(args);
runRecursive();
}
}
这样就完成了手写CPS的过程。这个过程中蕴含一些重要的思想。例如函数调用的地方不实际调用,而只是产生描述调用的数据,通过另一个执行函数去执行实际的函数调用。这个过程事实上是一个生成指令数据/解释指令执行的过程。这个思想在任何需要的地方都是一个不错的选择。
控制结构(11): Continuation passing style(CPS)的更多相关文章
- feilong's blog | 目录
每次把新博客的链接分享到技术群里,我常常会附带一句:蚂蚁搬家.事实上也确实如此,坚持1篇1篇的把自己做过.思考过.阅读过.使用过的技术和教育相关的知识.方法.随笔.索引记录下来,并持续去改进它们,希望 ...
- scheme Continuation
Continuation Pass Style在函数式编程(FP)中有一种被称为Continuation Passing Style(CPS)的风格.在这种风格的背后所蕴含的思想就是将处理中可变的一部 ...
- 尾递归(Tail Recursion)和Continuation
递归: 就是函数调用自己. func() { foo(); func(); bar(); } 尾调用:就是在函数的最后,调用函数(包括自己). foo(){ return bar(); } 尾递归:就 ...
- Haskell语言学习笔记(29)CPS
CPS (Continuation Passing Style) CPS(延续传递风格)是指函数不把处理结果作为返回值返回而是把处理结果传递给下一个函数的编码风格. 与此相对,函数把处理结果作为返回值 ...
- 【理论面试篇】收集整理来自网络上的一些常见的 经典前端、H5面试题 Web前端开发面试题
##2017.10.30收集 面试技巧 5.1 面试形式 1) 一般而言,小公司做笔试题:大公司面谈项目经验:做地图的一定考算法 2) 面试官喜欢什么样的人 ü 技术好. ...
- JavaScript内存优化
JavaScript内存优化 相对C/C++ 而言,我们所用的JavaScript 在内存这一方面的处理已经让我们在开发中更注重业务逻辑的编写.但是随着业务的不断复杂化,单页面应用.移动HTML5 应 ...
- jQuery中的Deferred-详解和使用
首先,为什么要使用Deferred? 先来看一段AJAX的代码: var data; $.get('api/data', function(resp) { data = resp.data; }); ...
- Understanding continuations
原文地址http://fsharpforfunandprofit.com/posts/computation-expressions-continuations/ 上一篇中我们看到复杂代码是如何通过使 ...
- 【转】对 Go 语言的综合评价
以前写过一些对 Go 语言的负面评价.现在看来,虽然那些评价大部分属实,然而却由于言辞激烈,没有点明具体问题,难以让某些人信服.在经过几个月实际使用 Go 来构造网站之后,我觉得现在是时候对它作一些更 ...
随机推荐
- 改善 C# 的语言习惯(一) - 使用属性而不是可访问的数据成员(整理中)
改善 C# 的语言习惯(一) - 使用属性而不是可访问的数据成员 序 为什么我们的程序运行得棒棒的,还要改呢?Why? 答:我们要让程序运行得更快,执行的效率更高,代码的可读性更强,维护的成本更低.. ...
- [MySQL] 索引中的b树索引
1.索引如果没有特别指明类型,一般是说b树索引,b树索引使用b树数据结构存储数据,实际上很多存储引擎使用的是b+树,每一个叶子节点都包含指向下一个叶子节点的指针,从而方便叶子节点的范围遍历 2.底层的 ...
- Android Studio 学习(七)通知
导入support- -v4 1.进入 file-project structure 2.左边选择app 3.右边选择dependencies 4.左下角可以看到一个加号,点击选择Library de ...
- webpack入门教程--3
webpack打包还可以使用配置文件,我们先创建一个叫做webpack.config.js的文件.这里需要注意一下,这个JS文件的名字不是我们胡乱写的,也是不能更改的,因为webpack 命令执行后, ...
- springboot之JdbcTemplate
springboot可以使用JdbcTemplate进行数据库访问,代码如下 添加pom文件 <parent> <groupId>org.springframework.boo ...
- Linux驱动学习1.hello world;
最近项目需要使用Linux系统开发,借此机会学习一下Linux驱动开发 hello word代码hello.c #include <linux/module.h> #include < ...
- Selenium自动化测试-进阶2-框架篇
前面的文章已经讲述了 Selenium自动化的入门知识,不知道各位看官看懂没有,有不懂的请留言. 接下来,开始讲解 Selenium自动化测试进阶知识. 首先讲解:自动化测试的核心: 利用自动化程序代 ...
- prufer序列笔记
prufer序列 度娘的定义 Prufer数列是无根树的一种数列.在组合数学中,Prufer数列由有一个对于顶点标过号的树转化来的数列,点数为n的树转化来的Prufer数列长度为n-2. 对于一棵确定 ...
- 逛csdn看见的一个知识阶梯,感觉不错
逛csdn看见的一个知识阶梯,感觉不错: 计算机组成原理 → DOS命令 → 汇编语言 → C语言(不包括C++).代码书写规范 → 数据结构.编译原理.操作系统 → 计算机网络.数据库原理.正则表 ...
- 最简单打开三星note8三星galaxy susb调试模式的方法
每当我们使用安卓手机连接PC的时候,如果手机没有开启usb调试模式,PC则无办法成功检测到我们的手机,部分APP也无办法正常使用,这时我们需要找处理方法将手机的usb调试模式开启,以下内容我们介绍三星 ...