企业🤖AI Agent构建引擎,智能编排和调试,一键部署,支持私有化部署方案 广告
附录B 在DOS中运行Scheme脚本 ========================== DOS的脚本也叫做“批处理”。通常一个输出`Hello World!`的DOS批处理文件应该这样写: ```shell echo Hello, World! ``` 这里用到了DOS的命令`echo`。脚本文件被命名为`hello.bat`,这样会被操作系统认为是可执行的。然后可以放入任一在`PATH`环境变量中的目录下。然后任何时候在DOS提示符下输入 ```shell hello.bat ``` 或者更简单的: ```shell hello ``` 就能立即得到这句俗得不能再俗的问候。 Scheme版本的`hello`批处理文件会用Scheme产生相同的输出,但是我们需要在文件中写一些东西来告诉DOS让它用Scheme来分析文件内容,而不是理解为它默认的脚本批处理语言。Scheme的批处理文件(也命名为`hello.bat`),内容如下: ```bat ;@echo off ;goto :start #| :start echo. > c:\_temp.scm echo (load (find-executable-path "hello.bat" >> c:\_temp.scm echo "hello.bat")) >> c:\_temp.scm mzscheme -r c:\_temp.scm %1 %2 %3 %4 %5 %6 %7 %8 %9 goto :eof |# (display "Hello, World!") (newline) ;:eof ``` 到`|#`之前全部是标准的DOS批处理命令。后面是Scheme的问候代码。最后还有一行标准的DOS批处理,即`;:eof`。 当用户在DOS提示符下输入`hello`时,DOS读取并将`hello.bat`作为一个普通的批处理文件来运行。第一行,`;@echo off`,关闭了命令运行时产生的输出――我们不想让大量冗余信息淹没我们脚本产生的输出。第二行,`;goto :start`,让脚本的执行跳转到标号`:start`即第四行。后面接着的`echo`行创建了一个叫`c:\_temp.scm`的Scheme临时文件,其内容如下: ```scheme (load (find-executable-path "hello.bat" "hello.bat")) ``` 下一个批处理命令调用MzScheme。`-r`选项加载Scheme文件`c:\_temp.scm`。所有的参数(在这个例子里没有)可以在Scheme中通过`argv`向量来获得。这个调用的Scheme会对我们的脚本进行求值,我们下面会看到。Scheme执行返回后,我们仍然需要让批处理文件正常地结束(否则就会碰到它不认识的Scheme代码了)。下一个批处理命令是`goto :eof`,这会让控制流跳过所有的Scheme代码,到达文件末尾,也就是包含`;:eof`标签的一行。然后脚本结束运行。 现在我们知道如何让Scheme来执行它的那部分代码,即运行嵌入在批处理文件中的Scheme表达式。载入`c:\_temp.scm`会使Scheme找到`hello.bat`文件的绝对路径(用`find-executable-path`过程),然后载入`hello.bat`。 因此,Scheme脚本文件现在会以Scheme文件来运行,文件中的Scheme表达式可以通过向量`argv`来访问脚本的原始参数。 现在Scheme略过脚本中的批处理命令。这是因为这些批处理命令要么以分号开头,要么用`#|...|#`包裹,这在Scheme看来是注释。 文件剩下的部分当然是Scheme代码,因此表达式被依次求值(最后的表达式`;:eof`是一个Scheme注释,因此没有关系)总之所有的表达式被求值后,Scheme会退出。 综上所述,在DOS提示符下输入hello会产生 ``` Hello, World! ``` 并返回DOS提示符。