控制结构(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 来构造网站之后,我觉得现在是时候对它作一些更 ...
随机推荐
- Haskell复习笔记(二)
Haskell中的递归 递归就是定义函数以调用自身的方式,关于递归解决问题的实例有很多,如斐波那契数列,还有汉诺塔问题,递归也正是Haskell中用来解决循环问题的关键. 自定义maxinum函数 m ...
- 杭电ACM2007--平方和与立方和
平方和与立方和 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Sub ...
- python之字符串反转
def main(): a = "abcdefg" a = a[::-1] print(a) if __name__ == '__main__': main()
- 测试一波SpringBoot的HTTP吞吐量
本来,其实就我个人而言现在很少去弄性能这一块的阵地了,主要在做设计与架构,不过前几天刚刚关注公众号的罗哥给我抛了关于性能方面的问题. 一个问题立马引起了我的兴趣,太久没弄性能方面的事情了,所以在隔天有 ...
- js正则表达式 数字和小数点 非负数 保留两位小数点
验证数字非负数 小数点保留两位小数点 下面正则已验证通过 /^(?!0+(?:\.0+)?$)(?:[1-9]\d*|0)(?:\.\d{1,2})?$/
- springboot之多数据源配置JdbcTemplate
springboot多数据源配置,代码如下 DataSourceConfig package com.rookie.bigdata.config; import org.springframework ...
- USSD 杂记
Android Oreo允许应用程序读取来自运营商的USSD消息. 利用emoney执行话费充值,需要执行USSD代码,尝试编写apk执行ussd代码进行充值. 尝试在Android8的系统上进行US ...
- 阿里云 API调用实践(python语言)
1.结论:阿里云的SDK开发,其实就是远程调用API,python的代码就是一个外壳,核心是封装成一个http报文,利用json格式,进行RPC调用. 2.SDK调用API的套路如下: # -*- c ...
- 解决ViewGroup不调用onDraw()的问题
今天在做项目的时候自定义了一个View,继承了LinearLayout,结果,里面的onDraw()方法一直无法被调用. 后来发现ViewGroup是默认不调用onDraw()方法的. 原因我们暂且不 ...
- 【原】Java学习笔记001 - JAVA开发环境搭建
1.JDK下载并安装,以jdk-7u45-windows-i586.exe为例(注意JDK的安装和JRE的安装是分开的) 2.“我的电脑”右键属性,找到“高级系统设置”,找到“高级”tab下的“环境变 ...