第七章 输入输出
===============
Scheme的输入/输出程序可以使你从输入端口读取或者将写入到输出端口。端口可以关联到控制台,文件和字符串。
## 7.1 读取
Scheme的读取程序带有一个可选的输入端口参数。如果端口没有特别指定,则假设为当前端口(一般是控制台)。
读取的内容可以是一个字符,一行数据或是S表达式。当每次执行读取时,端口的状态就会改变,因此下一次就会读取当前已读取内容后面的内容。如果没有更多的内容可读,读取程序将返回一个特殊的数据——文件结束符或EOF对象。这个对象只能用`eof-object?`函数来判断。
`read-char`程序会从端口读取下一个字符。`read-line`程序会读取下一行数据,并返回一个字符串(不包括最后的换行符),`read`程序则会读取下一个S表达式。
## 7.2 写入
Scheme的写入程序接受一个要被写入的对象和一个可选的输出端口参数。如果未指定端口,则假设为当前端口(一般为控制台)。
写入的对象可以是字符或是S表达式。
`write-char`程序可以向输出端口写入一个给定的字符(不包括`#\`)。
`write`和`display`程序都可以向端口写入一个给定的S表达式,唯一的区别是:`write`程序会使用机器可读型的格式而`display`程序却不用。例如,`write`用双引号表示字符串,用`#\`句法表示字符,但`display`却不这么做。
`newline`程序会在输出端口输出一个换行符。
## 7.3 文件端口
如果端口是标准的输入和输出端口,Scheme的I/O程序就不需要端口参数。但是,如果你明确需要这些端口,则`current-input-port`和`current-output-port`这些零参数程序会提供这个功能,例如:
```scheme
(display 9)
(display 9 (current-output-port))
```
拥有相同的效果。
一个端口通过打开文件和这个文件关联在一起。`open-input-file`程序会接受一个文件名作为参数并返回一个和这个文件关联的新的输入端口。`open-output-file`程序会接受一个文件名作为参数并返回一个和这个文件关联的新的输出端口。如果打开一个不存在的输入文件,或者打开一个已经存在的输出文件,程序都会出错。
当你已经在一个端口执行完输入或输出后,你需要使用`close-input-port`或`close-output-port`程序将它关闭。
在下述例子中,假如文件`hello.txt`文件只包含一个单词`hello`。
```scheme
(define i (open-input-file "hello.txt"))
(read-char i)
=> #\h
(define j (read i))
j
=> ello
```
假如文件`greeting.txt`在下述程序运行前不存在:
```scheme
(define o (open-output-file "greeting.txt"))
(display "hello" o)
(write-char #\space o)
(display 'world o)
(newline o)
(close-output-port o)
```
现在`Greeting.txt`文件将会包含这样一行:
```
hello world
```
### 7.3.1 文件端口的自动打开和关闭
Scheme提供了`call-with-input-file`和`call-with-output-file`过程,这些过程会照顾好打开的端口并在你使用完后将端口关闭。
`call-with-input-file`程序接受一个文件名参数和一个过程。这个过程被应用在一个已打开的文件输入端口。当程序结束时,它的结果会在保证端口关闭后返回。
```scheme
(call-with-input-file "hello.txt"
(lambda (i)
(let* ((a (read-char i))
(b (read-char i))
(c (read-char i)))
(list a b c))))
=> (#\h #\e #\l)
```
`call-with-output-file`程序会对输出文件提供类似的服务。
## 7.4 字符串端口
一般来说将字符串与端口相关联是很方便的。因此,`open-input-string`程序将一个给定的字符串和一个端口关联起来。读取这个端口的程序将读出下述字符串:
```scheme
(define i (open-input-string "hello world"))
(read-char i)
=> #\h
(read i)
=> ello
(read i)
=> world
```
`open-output-string`创建了一个输出端口,最终可以用于创建一个字符串:
```scheme
(define o (open-output-string))
(write 'hello o)
(write-char #\, o)
(display " " o)
(display "world" o)
```
现在你可以使用`get-output-string`程序得到保留在字符串端口`o`中的字符串:
```scheme
(get-output-string o)
=> "hello, world"
```
字符串端口不需要显式地去关闭。
## 7.5 加载文件
我们已将看到`load`程序可以加载包含`Scheme`代码的文件。`load`一个文件意味着按顺序求值文件中每一个Scheme表达式。`load`中的路径参数是相对当前Scheme工作目录计算的,该工作目录一般是调用Scheme可执行文件时的目录。
一个文件可以加载其他的文件,这在包含许多文件的大项目中十分有用。但是,除非使用绝对路径,否则`load`参数中的文件位置将依赖于执行Scheme的当前目录。而提供绝对路径名并不是很方便,因为我们更愿意把项目文件作为一个单元(保留它们的相对路径名)在很多不同机器中运行。
Mzscheme提供了`load-relative`程序,可以很好的解决这个问题。`load-relative`,和`load`相似,带有一个路径名参数。当在`foo.scm`文件中出现`load-relative`调用时,它的参数的路径将根据文件`foo.scm`所在目录的路径来计算。特别注意的是,这个路径名和执行Scheme的当前目录无关,因此也就可以方便地进行多文件程序的开发。