lisp语法分析与执行分离

《sicp》4.1.7节读书笔记

这节中将书中之前实现的解释器做了改进,将语法分析于执行分离.这样做能提高很多效率,比如我们需定义一个过程,然后执行该过程多次.分情况说明:

1.语法分析于执行未分离

每次执行都会进行语法分析

2.语法分析于执行分离.

在定义该过程时,已经完成语法分析,该过程名保存在环境中的相应约束

就是,语法分析生成的(lambda (env) …).所以当执行该过程时,只需传入相应环境就可以完成执行,这就实现了一次分析,供永久执行.

实际例子:

输入语句

(define (f x) (+ x x))
(f 4)

分别分析以上两种情况的执行过程


//lisp解释器的语法分析和执行未分离的情况 输入: (define (f x) (+ x x)) ==> (eval ('define ('f 'x) ('+ 'x 'x)) env)
==> (eval-definition ('define ('f 'x) ('+ 'x 'x)) env )
==> (define-variable! (definition-variable ('define ('f 'x) ('+ 'x 'x)))
(eval (definition-value ('define ('f 'x) ('+ 'x 'x))) env)
env)
;其中的 (definition-variable ('define ('f 'x) ('+ 'x 'x)) ) 返回为 'f
(eval (definition-value ('define ('f 'x) ('+ 'x 'x))) env) 返回为 ('procedure ('x) ('+ 'x 'x) env)
//所以在这里执行完毕后 env环境中,多了一条f函数的约束.
('f ('procedure ('x) ('+ 'x 'x) env) env ) 输入: (f 4) ==> (eval ('f 4) env)
==> (apply (eval 'f env) 4)
;其中的(eval 'f env) 返回 ('procedure ('x) ('+ 'x 'x) env) ==>(apply ('procedure ('x) (+ 'x 'x) env) 4)
==>(eval-sequence ('+ 'x 'x) (('x:4) (env)) )
;其中(('x:4) (env)) 是执行(f 4)后创建的新环境
==>(eval ('+ 'x 'x) (('x:4) (env)))
==>(apply (eval '+ (('x:4) (env))) (4 4))
;其中(eval '+ (('x:4) (env))) 返回 ('primitive +) (假设初始环境含有 ""+"" 基本过程)
==>(apply ('primitive +) (4 4))
==>(apply-primitive-procedure ('primitive) (4 4)) ==>(apply-in-underlying-scheme
+ (4 4))
==>8
如果再输入一次(f 4)仍然将执行以上分析和执行过程,可见每执行次(f 4)解释器都将分析一次 f 函数,然后执行 //lisp解释器的语法分析与执行分开的情况 输入 (define (f x) (+ x x)) ==>(eval ('define ('f 'x) ('+ 'x 'x) begin_env)
==>((analyze ('define ('f 'x) ('+ 'x 'x))) begin_env)
;;;执行 (analyze ('define ('f 'x) ('+ 'x 'x)))
==>(analyze ('define ('f 'x) ('+ 'x 'x)) )
==>(analyze-definition ('define ('f 'x) ('+ 'x 'x)))
{
(let ((var 'f)
(vproc (analyze ('lambda ('x 'x) ('+ 'x 'x))))
(lambda (env)
(define-variable! var (vproc env) env)
'ok))
}
;;;执行 (vproc (analyze ('lambda ('x 'x) ('+ 'x 'x)))))
==>(analyze('lambda ('x 'x) ('+ 'x 'x)))
==>(analyze-lambda ('lambda ('x 'x) ('+ 'x 'x)))
{
(let ((vars ('x 'x))
( bproc (analyze-sequence ( ('+ 'x 'x)) )))
(lambda (env) (make-procedure ('x 'x) brpoc env)))
}
;;执行bproc
==>(analyze-sequence (('+ 'x 'x)))
==>{ (analyze ('+ 'x 'x))
(analyze-application ('+ x x)){
(let ((fproc (analyze '+))
(aprocs (map analyze ('x 'x))))
(lambda (env)
(execute-application (fproc env)
(map (lambda (aproc) (aproc env) aprocs))))
}
;执行(analyze '+)
==>(analyze-variable '+)
返回 :(lambda (env) (lookup-variable-value '+ env))
;执行(map analyze ('x 'x))
返回 :((lambda (env) (look-variable-value 'x env)) (lambda(env) (look-variable-value 'x env)))
返回:bproc:
(lambda(env)
(execute-application ((lambda (env) (lookup-variable-value '+ env)) env)
(map (lambda (aproc) (aproc env)) ((lambda (env) (look-variable-value 'x env)) (lambda(env) (look-variable-value 'x env))))
)) }}
==>返回:vproc=
(lambda (env) (make-procedure ('x 'x) brpoc env))
==>(make-procedure ('x 'x) brpoc env)
==>返回:('procedure ('x 'x) brpoc env)
==>返回:
(lambda(env)
(define-varibale! 'f) (vproc env) env)
;;代入 begin_env
==>((lambda(env) (define-varibale! 'f (vproc env) env) begin_env))
==>(define-varibale! 'f (vproc begin_env) begin_env)
==>(vproc begin_env)
f : ('procedure ('x 'x) brpoc begin_env)
最终:在 begin_env 中
f=('procedure
('x 'x)
(lambda(env)
(execute-application
((lambda (env) (lookup-variable-value '+ env)) env)
(map (lambda (aproc) (aproc env)) ((lambda (env) (look-variable-value 'x env)) (lambda(env) (look-variable-value 'x env))))))) //输入 (f 4)
==>(eval ('f 4) changed_env)
==>((analyze ('f 4)) changed_env)
==>(analyze-application ('f 4))
==>{
(let ( (fproc (lambda(env) (lookup-varible-value 'f env)))
(approcs (lambda(env) 4))
(lambda(env)
(execute-application (fproc env)
(map (lambda (aproc) (aproc env)) aproc)))))
}
==>代入changed_env
==> (execute-application (fproc changed_env) 4)
==> (execute-application
('procedure
('x 'x)
(lambda(env)
(execute-application
((lambda (env) (lookup-variable-value '+ env)) env)
(map (lambda (aproc) (aproc env)) ((lambda (env) (look-variable-value 'x env)) (lambda(env) (look-variable-value 'x env)))))))
4)
==>((lambda(env)
(execute-application
((lambda (env) (lookup-variable-value '+ env)) env)
(map (lambda (aproc) (aproc env)) ((lambda (env) (look-variable-value 'x env)) (lambda(env) (look-variable-value 'x env))))))) ('x:4 (begin_env)))
==>(execute-application ('primitive +) (4 4))
==>(apply-in-underlying-scheme
+ (4 4))
==>8

lisp语法分析与执行分离的更多相关文章

  1. 语法分析器初步学习——LISP语法分析

    语法分析器初步学习——LISP语法分析 本文参考自vczh的<如何手写语法分析器>. LISP的表达式是按照前缀的形式写的,比如(1+2)*(3+4)在LISP中会写成(*(+ 1 2)( ...

  2. Js引擎解析执行 阅读笔记

    Js引擎解析执行 阅读笔记 一篇阅读笔记 http://km.oa.com/group/2178/articles/show/145691?kmref=search&from_page=1&a ...

  3. 第1章列表处理——1.1 Lisp列表

    Lisp是啥? Lots of Isolated Silly Parentheses (大量分离的愚蠢的括号) Lisp指的是"LISt Processing"(列表处理),通过把 ...

  4. sqlserver 学习之分离与附加数据库

    在学习sqlserver数据库的过程中,我们会学习到有关一些听起来比较陌生的专用名词,比如说分离与附加这两个专有名词,对于我来说就是比较陌生的.好的,下面我们一起来学习一下吧.为了讲的通俗一点,下面以 ...

  5. [转帖]linux screen 命令详解,xshell关掉窗口或者断开连接,查看断开前执行的命令

    linux screen 命令详解,xshell关掉窗口或者断开连接,查看断开前执行的命令 https://binwaer.com/post/12.html yun install -y screen ...

  6. ORACLE数据库学习之体系结构

     Oracle体系结构 ORACLE数据库体系结构决定了oracle如何使用网络.磁盘和内存.包括实例(instance),文件(file)和进程(process不包括后台进程)三部分. 实例:每 ...

  7. ANTLR4权威指南 - 第7章 通过特定应用程序代码解耦语法

    第7章 通过特定应用程序代码解耦语法 到目前为止,我们已经知道了怎么用ANTLR的语法来定义语言了,接下来我们要给我们的语法注入一些新的元素了.就语法本身而言,其用处并不大,因为它只能告诉我们一个用户 ...

  8. Oracle 后台进程 详细说明

    一. 进程概述 先来看一下Oracle 11g 的架构图. 看起来比较模糊,我已经上传到了csdn 的下载. 是个pdf 文件, 2m 多. 那个看起来比较清楚. 也对每个进程做了解释. 下载地址:O ...

  9. Golang的演化历程

    本文来自Google的Golang语言设计者之一Rob Pike大神在GopherCon2014大会上的开幕主题演讲资料“Hello, Gophers!”.Rob大神在这次分 享中用了两个生动的例子讲 ...

随机推荐

  1. MySQL连接查询流程源码

    http://blog.itpub.net/29510932/viewspace-2129300/ 初始化: 点击(此处)折叠或打开 main |-mysqld |-my_init // 初始话线程变 ...

  2. 打造kubernetes 高可用集群(nginx+keepalived)

    一.添加master 部署高可用k8s架构 1.拷贝/opt/kubernetes目录到新的master上(注意如果新机上部署了etcd要排除掉) scp -r /opt/kubernetes/ ro ...

  3. 190919 centos系统中python2卸载重装

    问题:某些原因卸载了python2,连带卸载了yum工具. 解决思路: 如果服务器没有什么东西,重装系统最省事.但是如果不允许重装,那就只能按部就班的恢复python2和yum. 步骤: 删除pyth ...

  4. 使用Cloudera Manager搭建zookeeper集群及HDFS HA实战篇

    使用Cloudera Manager搭建zookeeper集群及HDFS HA实战篇 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.使用Cloudera Manager搭建zo ...

  5. 新添加的磁盘大于2T 的分区方法

    环境CentOS7.1 2.9t磁盘 fdisk 只能分区小于2t的磁盘,大于2t的话,就要用到parted 1,将磁盘上原有的分区删除掉: 进入:#parted   /dev/sdb 查看:(par ...

  6. amazeui datepicker日历控件 设置默认当日

    amazeui datepicker日历控件 设置默认当日 背景: 最近在做一个系统的时候,前台需要选择日期,传给后台进行处理,每次都需要通过手动点击组件,选择日期,这样子很不好,所以我想通过程序自动 ...

  7. Integer Inquiry UVA-424(大整数)

    题意分析: 将字符串倒着存入int数组中,每次加完后再取余除去大于10的部分 关键:倒着存入,这样会明显缩短代码量. #include<iostream> #include<cstd ...

  8. Kotlin伴生对象及其字节码内幕详解

    继续面向对象,开撸就是!! 接口: 我们知道对于JDK8之后接口中除了方法的声明之后还可以有default方法的,而在Kotlin中也类似,下面来看一下在Kotlin接口相关的东东: 很显然就是一个方 ...

  9. Spring4 -03 -Dependency Injection (依赖注入) : 代码体现/配置xml/测试

    DI:中文名称:依赖注入 英文名称((Dependency Injection) DI 是什么? 3.1 DI 和IoC 是一样的,差不多一样的技术和模板! 3.2 当一个类(A)中需要依赖另一个类( ...

  10. 04 c++中的友元

    c++中的类具有封装性,类中的私有数据只有该类的成员函数可以访问,程序中访问类中的私有成员,必须通过对象来调用成员函数,但是频繁的调用会使运行效率降低. 为了解决上述问题,c++中加入友元机制,友元可 ...