控制结构(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# 爬虫----Cookies处理(Set-Cookie)
/// <summary> /// Cookie 助手 /// </summary> public class CookieHelper { /// <summary&g ...
- 零基础学Python--------进阶篇 第6章 函数
第6章 函数 6.1函数的创建和调用 提到函数,大家会想到数学函数吧,函数是数学最重要的一个模块,贯穿整个数学学习过程.在Python中,函数的应用非常广泛.在前面我们已经多次接触过函数.例如,用于输 ...
- 安装屏保软件(Linux终端演示 “黑客帝国” 字母雨界面)和Linux修改管理员密码
1.Linux修改管理员密码:打开终端:1. 重启 reboot 2.进入内核登陆系统点击e3.进入系统救援界面,定位Linux16所在行,找到ro 后删除,在此位置添加一条命令: rw init= ...
- 从零学习Fluter(七):Flutter打包apk详解
写一个win上 flutter 打包apk的教程 这篇文档介绍一下flutter打包发布正式版apk 整体来看,和命令行打包rn的方法相差不大 打包前先做检查工作&查看构建配置 Android ...
- 关于JPasswordField的getText()方法过时问题解决
这几天想做一个登陆界面,用Jframe做,连接数据库时发现JPasswordField的getText()过时了,没法使用.查了资料发现改成了: try{ String sql="SELEC ...
- Spring MVC 数据绑定 (四)
完整的项目案例: springmvc.zip 目录 实例 项目结构路径: 一.配置web.xml <?xml version="1.0" encoding="UTF ...
- System.map文件的作用
有关System.map文件的信息好象很缺乏.其实它一点也不神秘,并且在整个事情当中它并不象看上去那么得重要.但是由于缺乏必要的文档说明,使其显得比较神秘.它就象耳垂,我们每个人都有,但却不知道是干什 ...
- 基于MFC的学生成绩管理系统的设计与实现
1.技术介绍MFC是微软基础类库的简称,是微软公司实现的一个C++类库,主要封装了大部分的WINDOWS API函数,并且包含一个应用程序框架,以减少应用程序开发人员工作量.VC++是微软公司开发的C ...
- 微信js-sdk开发获取签名和获取地理位置接口示例
###微信js-sdk开发获取签名和获取地理位置接口示例 前言:在做微信公众号开发时需要获取用户的地理位置信息,之前通过高德或者百度.腾讯等地图的api时发现经常获取不到,毕竟第三方的东西,后来改为采 ...
- Spring Boot使用注解实现AOP
第一步: 添加依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId& ...