面向堆栈编程面向堆栈编程,或基于堆栈编程,是依赖于堆栈机器模型来操纵数据和传递参数的编程范型。一些编程语言适合这种描述,著名的有Forth、RPL、 PostScript、BibTeX样式设计语言[1]和很多汇编语言。 概述面向堆栈语言运算于一个或多个堆栈之上,每个都充任不同用途。因此,用其他编程语言构造的程序,在面向堆栈系统中使用可能需要修改[2]。进一步的说,一些面向堆栈语言运算,采用后缀表示法也称为逆波兰表示法,就是说,命令的任何实际参数(argument)或形式参数(parameter),都在这个命令之前陈述。例如,后缀表示法写为 基于堆栈算法PostScript是后缀式基于堆栈语言的例子。在这种语言中表达式的一个例子是 堆栈导向可以用如下传送带类比来体现。在传送带(输入)末端,按顺序摆放了标记着 拾取盘子 这是一个非常简单的计算。如果是更复杂的计算比如
在处理完所有输入之后,这个堆栈包含 从而可以得出如下结论:基于堆栈的编程语言只有一种处理数据方式,即通过从堆栈顶部选取一块数据,术语叫做“弹出”,和将数据放回堆栈,术语叫“压入”。可以按常规或用其他种类编程语言书写的任何表达式,可以写成后缀(或前缀)形式,从而服从面向堆栈语言去做出解释。 堆栈操纵因为堆栈是在面向堆栈语言中操纵数据的关键方式,这种语言通常提供某种堆栈操纵算子。经常提供的有: 堆栈作用的示意为了辅助理解语句的作用,使用简短注释来展示在这个语句前后的堆栈顶部。如果有多个项目,堆栈的顶部在最右端。这个表示法常用于Forth语言,在它那里的注释包围在圆括号内。 ( 之前 -- 之后 )
例如,描述如下基本Forth堆栈算子: dup ( a -- a a )
drop ( a -- )
swap ( a b -- b a )
over ( a b -- a b a )
rot ( a b c -- b c a )
还有如下这样描述 fib ( n -- n' )
它等价于霍尔逻辑的先决条件和后置条件。二者注释也可以称为断言,尽管在基于堆栈语言中无此必要。 PostScript堆栈PostScript和其他一些堆栈语言有用于其他用途的其他独立堆栈。 变量和字典已经分析了不同表达式的求值。变量的实现对于任何编程语言都是重要的,但是对于面向堆栈语言,它有着特殊关切,因为这是与数据交互的唯一方式。 在面向堆栈语言比如PostScript中实现变量的方式,通常涉及到一个独立的特殊化了堆栈,它持有键-值对的“字典”。要建立一个变量,首先必须建立一个键(变量名字),具有与之关联的一个值。在PostScript中,一个名字数据对象前缀着一个 /x 42 def
在堆栈顶部的字典中,将名字 过程在基于堆栈语言中,过程自身被当作数据对象。在PostScript中,过程被指示在 { dup mul }
表示一个匿名过程,它重复在堆栈顶部的东西并接着相乘二者得出结果,这是个求平方的过程。 因为过程被当作一个简单的数据对象,可以定义过程的名字。在检索到它们的时候,直接执行它们。字典在存储各种定义的同时,提供了控制作用域的一种方式。 因为数据对象被存储在最顶部字典,自然出现了一种未预料的能力:在从一个字典查找一个定义的时候,检查最顶部字典,接下一直往下。如果在一个不同的字典中已经定义了同名的一个过程,调用局部的那个过程。 典型过程的剖析过程经常接受实际参数。它们被过程以非常特殊的方式处理,不同于其他编程语言。下面查看PostScript下的一个斐波那契数列程序: /fib
{
dup dup 1 eq exch 0 eq or not
{
dup 1 sub fib
exch 2 sub fib
add
} if
} def
在堆栈上使用了递归定义。斐波那契数函数接受一个实际参数。首先,它被测试是否为 假定计算 堆栈:4 dup 堆栈:4 4 dup 堆栈:4 4 4 1 eq 堆栈:4 4 false exch 堆栈:4 false 4 0 eq 堆栈:4 false false or 堆栈:4 false not 堆栈:4 true 因为这个表达式求值为真,求值最内层过程: 堆栈:4 dup 堆栈:4 4 1 sub 堆栈:4 3 fib
堆栈:4 F(3) exch 堆栈:F(3) 4 2 sub 堆栈:F(3) 2 fib
堆栈:F(3) F(2) add 堆栈:F(3)+F(2) 这是预期的结果。 这个过程不使用命名变量,单纯使用堆栈。命名变量可以使用 堆栈:3 /n exch 堆栈:/n 3 def 堆栈:空(原内容已被用于定义) n 堆栈:3 n 堆栈:3 3 mul 堆栈:9 这是预期结果。 控制和流程因为存在匿名过程,流程控制可以自然出现。if…then…else语句需要三段数据:条件,如果条件为真要做的过程,和如果条件为假要做的过程。例如在PostScript中: 2 3 gt { (2 is greater than three) = } { (2 is not greater than three) = } ifelse
所进行的几乎等价于C代码: if (2 > 3) { printf("2 is greater than three\n"); } else { printf("2 is not greater than three\n"); }
循环和其他构造也是类似的。 基于堆栈编程语言参见引用
|