作者 | Brian Robert Callahan
编译 | 张洁 责编 | 屠敏
为了让挑战变得有意思,Brian Robert Callahan(纽约州北部伦斯勒理工学院信息技术和网络科学的讲师) 决定编写一个看起来一点也不像 C 的 C 程序。如果把 C 语言变成另一种语言,然后用那种语言去编写程序,那么使用 C 编译器编译程序会怎样?
能编写出以下程序,Brian Robert Callahan 觉得是一件很光辉的事。
var a, b, c, h, i, l, v, x, y, q, w, p size 65535 ,
packed n size 13 ꞉integer ;
procedure display(r,s,c) ;
begin
LOOP
call A(Z) ;
call H(y,x) ;
call B(Z)
POOL ;
y ꞉= r;
x ꞉= s;
call A(c) ;
call H(y,x) ;
call B(c) ;
call refresh()
end ;
procedure fill(y,x,c,a) ;
begin
if(y<0 or y>w-1 or x<0 or x>q-1 or c = a or Z <> a)fill꞉= -1 ;
call draw(c) ;
call fill(y+1,x,c,a) ;
call fill(y-1,x,c,a) ;
call fill(y,x-1,c,a) ;
call fill(y,x+1,c,a)
end ;
procedure save(r,s) ;
begin
i ꞉= 0 ;
while(i<13)do
begin
n[i] ꞉= 0 ;
i ꞉= i+1
end ;
call move(w>>1,(q>>1)-6) ;
call printw("Save: ") ;
call echo() ;
call getnstr(n,12) ;
call noecho() ;
call open(n,"w+") ;
call writeChar(83) ;
call writeChar(w) ;
call writeChar(q) ;
LOOP
call writeChar(Z)
POOL ;
y ꞉= r;
x ꞉= s;
call close ;
call move(y,x)
end ;
procedure load(packed ʌ n) ;
begin
call open(n,"r") ;
call check ;
LOOP
readln(c);
call draw(c)
POOL ;
c ꞉= 0;
call close
end ;
procedure m() ;
begin
l ꞉= 0 ;
v ꞉= 1 ;
call A(c) ;
call H(0,0) ;
call B(c) ;
call refresh() ;
while(v)do
begin
read(inp) ;
'/':l ꞉= not l ; if(l)call draw(c) ; OK
'k':y ꞉= y-1 ; if(y<0)y ꞉= 0 ; if(l)call draw(c) ; OK
'j':y ꞉= y+1 ; if(y>w-1)y ꞉= w-1 ; if(l)call draw(c) ; OK
'h':x ꞉= x-1 ; if(x<0)x ꞉= 0 ; if(l)call draw(c) ; OK
'l':x ꞉= x+1 ; if(x>q-1)x ꞉= q-1 ; if(l)call draw(c) ; OK
' ':call draw(c) ; OK
'c':c ꞉= c+1 ; if(c = M)c ꞉= 0 ; OK
'd':call draw(15) ; OK
'f':call fill(y,x,c,Z) ; OK
's':call save(y,x) ; OK
'q':v ꞉= 0 ; OK
'v':c ꞉= c-1 ; if(c = N)c ꞉= M-1 ; CALL display(y,x,c)
end
end ;
procedure main(I c,packed ʌ ʌ v) ;
begin
call start ;
call getmaxyx(stdscr,w,q) ;
if(w>M)w ꞉= M ;if(q>M)q ꞉= M ;
call start_color() ;
while(x<M)do
begin
call init_pair(x,x,x) ;
x ꞉= x+1
end ;
LOOP
call draw(15)
POOL ;
if(c = 2)call load(v[c-1]) ; call display(0,0,0) ; call m() ; call endwin()
end ;
call main.
通过这段代码,也许有很多网友直言,这并非是 C 代码,而是包含 Pascal 语言的所有特征,用分号作为语句分隔符而不是语句结束符,用“:=”表示赋值,也许还有一些 Algol(一种指令式编程语言)的 LOOP……POOL 语法。它甚至有 Pascal 语言的返回赋值,即给函数分配一个值,就是它的返回值。
不过,Brian Robert Callahan 对此解释道,最近听说 A+、K 和 Q 语言(它们是 APL 和 J 之类的数组编程语言)的开发者 Arthur Whitney 会使用 C 预处理器来创建自己的语言,然后用这种自定的语言来实现他的语言。于是,他决定尝试一下这个有趣的实验。
Brian Robert Callahan 基于 PL/0 的自定义语言,编写了一个编译器。这个练习的亮点是 C 编译器将 UTF-8 字符理解为标识符的有效字符。Brian Robert Callahan 使用了很多看起来像 ASCII(美国信息交换标准代码)的字符,实际上并不是 ASCII,最终还是被接受为有效的标识符。C 预处理器能很好地将这些标识符转化为相对应的指令。在这里可以看到隐藏起来的头文件。
如果想查看 C 代码的真实外观,请尝试运行:
$ cc -E cpaint.c | clang-format | less
如果你感兴趣的话,不妨去试试上面的程序吧!
原文链接:https://briancallahan.net/blog/20220220.html
《新程序员003》正式上市,50余位技术专家共同创作,云原生和数字化的开发者们的一本技术精选图书。内容既有发展趋势及方法论结构,华为、阿里、字节跳动、网易、快手、微软、亚马逊、英特尔、西门子、施耐德等30多家知名公司云原生和数字化一手实战经验!
☞