庄周梦蝶

生活、程序、未来
   :: 首页 ::  ::  :: 聚合  :: 管理

应用序 or 正则序?

Posted on 2007-05-08 15:11 dennis 阅读(4313) 评论(6)  编辑  收藏 所属分类: 计算机科学与基础
    这是《计算机程序的构造与解释》中的一道习题,如何去判断一个scheme解释器是采用什么方式进行求值的?应用序 or 正则序。应用序是先对参数求值而后应用,而正则序则相反——完全展开而后归约求值。正则序相比于应用序,会部分存在重复求值的情况。习题是这样的:
    Ben Bitdiddle发明了一种检测方法,能够确定解释器究竟采用的哪种序求值,是采用正则序,还是采用应用序,他定义了下面两个过程:
  
(define (p) (p))
(define (test x y)
    (
if (= x 0)
         
0
         y))
    而后他求值下列的表达式:
  
(test 0 (p))
如果解释器采用的是应用序求值,ben将会看到什么情况?如果是正则序呢?

    分别分析下这两种情况下解释器的求值过程:
1.如果解释器是应用序,将先对过程test的参数求值,0仍然是0,(p)返回的仍然是(p),并且将无穷递归下去直到栈溢出,显然,在这种情况下,解释器将进入假死状态没有输出。

2.如果解释器是正则序,完全展开test过程:
(define (test 0 (p))
    (
if (= 0 0)
        
0
        (p))
接下来再进行求值,显然0=0,结果将返回0。

    一般lisp的解释器都是采用应用序进行求值。这个问题在习题1.6中再次出现。我们知道scheme已经有一个cond else的特殊形式,为什么还需要一个if else的特殊形式呢?那么我们改写一个new-if看看:
(define (new-if predicate then-clause else-clause)
        (cond (predicate then
-clause)
              (
else else-clause)))

写几个过程测试一下:
(new-if (< 1 01 0)
结果一切正常,但是,当这3个参数是过程的时候会发生什么情况呢?在这3个参数如果存在递归调用等情况下,解释器也将陷入无限循环导致栈溢出!比如书中的求平方根过程用new-if改写:
(define (new-if predicate then-clause else-clause)
        (cond (predicate then
-clause)
              (
else else-clause)))
(define (average x y)(
/ (+ x y) 2))
(define (square x) (
* x x))
(define (improve guess x)(average guess (
/ x guess)))
(define (good_enough
? guess x)
        (
< (abs (- (square guess) x)) 0.000001))
(define (sqrt_iter guess x)
        (new
-if (good_enough? guess x)
            guess
            (sqrt_iter (improve guess x) x)))   
(define (simple_sqrt x)(sqrt_iter 
1 x))

因为解释器是应用序求值,将对new-if过程的3个参数求值,其中第三个参数也是一个过程(sqrt_iter (improve guess x) x)) 递归调用自身,导致无限循环直到栈溢出。




评论

# re: 应用序 or 正则序?  回复  更多评论   

2008-09-22 22:45 by stephanxu@sina.com
谢谢解答,脑瓜好久不用,锈掉了

# re: 应用序 or 正则序?  回复  更多评论   

2010-12-02 17:40 by baozii
我觉得第二题和正则序&应用序没有关系呢……

即使是正则序也是会陷入死循环的

解释器这样做的原因或许是因为三者都是作为函数的参数传递进去的,而解释器会对每个函数的参数进行求值

# re: 应用序 or 正则序?[未登录]  回复  更多评论   

2011-02-28 22:45 by 咸鱼
= =我还是分不清正则序跟应用序啊!!!!!!!

# re: 应用序 or 正则序?  回复  更多评论   

2013-03-10 00:53 by Victorique
http://codepad.org/BDtSL4nR
刚刚在看这一节,博主的文很有用,谢谢。
呵呵看来codepad为应用序咯。

# re: 应用序 or 正则序?  回复  更多评论   

2014-06-09 17:52 by Muout
@baozii
确实,第一题是应用序存在的问题,但是第二道题不管正则序还是应用序,都会有这个问题;
正则序完全展开时,会忽略条件,一直展开只至展开到基本元素,所以遇到由条件判断结束的递归就没法结束,一直在展开;
应用序在调用用户定义的普通过程时,会先计算所有的入参,不管这个入参会不会被使用到或者调用到,所以在递归中,也是因为计算值的需要一直在展开;

所以,不管正则序还是应用序都需要特殊处理的if语句;

# re: 应用序 or 正则序?[未登录]  回复  更多评论   

2015-01-05 17:41 by XXX
@baozii
第一题应用序,正则序都要跪!
我也做了一个测试
(define (add x) (if (< x 5000000) (add x) -5))
然后是
(define (mytest x y) (if (= x 0) 0 (add y)))
然后运行
(mytest 0 1)
结果瞬间出现了0,证明没有展开后面的(add y)
但是运行(test 0 (p))的时候,跪了

所以结论是,不管是正则序还是应用序,都要跪!

只有注册用户登录后才能发表评论。


网站导航: