当今最普遍的 roff 系统是自由软件的实现 GNU roff, groff(1). groff 之前的实现被称为传统的 classical 实现 (从 1973 年开始)。 groff 实现了它的传统的前辈的用法和功能,并且加入了更多扩展。 当前,由于 groff 是唯一可以在 (几乎是) 任何计算机系统上都可用的 roff 系统,因此它是事实上的 roff 标准。
在一些古老的 Unix 系统中,有一个叫做 roff 的可执行文件,实现了甚至更加古老的 Multics 操作系统中的 runoff 。 参见 HISTORY 段落。 它的功能非常有限,即使与古老的 troff 相比;它不再被支持了。 因此,在本文档中, roff 总是泛指 roff 系统 roff system
,不是古老的 roff 可执行文件
尽管历史悠久,roff 当前还在广泛使用中,例如,UNIX 系统的手册页 (man~pages/) ,很多软件书籍,系统文档,标准和团体组织的文档都是用 roff 来写的。 roff 在文本设备上的输出效果仍然是无可比拟的, 并且,与其他自由的排版系统相比,它的图形输出也不差, 甚至强于很多商业的系统
最普遍的 roff 的应用是手册页 manual pages (简称 man pages); 它是很多操作系统中的标准文档系统
此文档描述了围绕 roff system
开发的一些历史事件; 所有 roff 版本在用法方面的一些共同点, roff 管道的细节--它经常被掩盖在类似 groff(1) 等等的 “前端” 之后; 对排版语言的一般的概述; 一些写 roff 文档的技巧; 还有到更多资料的链接
roff 系统的演变与操作系统的历史紧密联系。 它的 “先祖” runoff 是 Jerry Saltzer 在 CTSS 操作系统 (Compatible Time Sharing System
,兼容分时系统 (?),1961) 上的作品。 CTTS 后来发展成为操作系统 Unix 的一个著名的来源,出现于 1963,同时 runoff 成为文档和文本处理的主要格式。 当时,这两种操作系统只能运行在非常昂贵的计算机之上, 因此它们大部分用于研究和官方及军队的任务之中
与现代的 roff 相比, runoff 语言可以做的事非常有限。 在 60 年代,只能产生文本的输出。 这可以用一个长度为~2 的命令 (request) 的集合实现, 这些命令的绝大部分都保持不变地被 roff 采用。 这种语言的模型是根据 “前计算机时代” 的排版习惯而建立的, 那时,以点 (dot, `.') 开头的行被写在手稿中, 向之后进行手工排版工作的工人指示格式化的要求
开始,runoff 程序是用 PL/1 语言写成的,后来用 BCPL
来写--那是 C~ 编程语言的 “祖母”。 在 Multics 操作系统中,帮助系统由 runoff 来处理, 与 roff 管理 Unix 手册页的作用类似。 仍然有 runoff 语言写成的文档, 例如可以到 Saltzer 的主页看看。参见 SEE ALSO
runoff 这个名字被简化为 roff. Ocsanna 所设想的,极大扩展了的语言已经包含了一个完整的 roff
系统的所有元素。 所有现代的 roff 系统都试图实现同这个系统兼容。 因此 Joe Osanna 是当之无愧的所有 roff 系统之父
最早的 roff 系统有三个排版程序
Osanna 的第一版用 PDP-11 的汇编语言实现,发布于 1973. Brian Kernighan 加入到 roff 的开发中,使用 C~编程语言将它进行了重写。 C~版本发布于 1975.
nroff/troff 程序使用的排版语言的语法记载在著名的 Troff User's Manual [CSTR~#54] 中,出版于 1976, Brian Kernighan 对它不断修订,直到 1992 年。 此文档是对传统的 classical troff
的说明. 所有后来的 roff 系统都试着与这个说明实现兼容
1977 年,Osanna 在他~50 岁时,由于一次突发的心脏病而去世。 Kernighan 继续开发 troff。 下一个开发目标是赋予 troff 一个更一般的接口,以支持更多设备, 中间输出格式和 “后处理” 系统。 这使得 roff 系统的结构趋于完整,现在仍然被使用。参见 USING ROFF
段。 1979 年,它们被写入论文 [CSTR~#97]. 这个新的 troff 版本是所有现存的较新的 troff 系统的基础,包括 groff. 在一些系统上,这个设备无关的 troff 有一个可执行文件叫做 ditroff(7). 所有现代的 troff 程序都已经自动提供了对 distroff 完整的兼容性
系统 ("their" system) 出售--- 只有很少的修改
古老的 Unix 和传统的 troff 的源代码在长达 20 年时间里不再可以自由获得。
幸运的是,Caldera 收购了 SCO UNIX (2001). 随后,Caldera 使得源代码可以在网上获得,允许用于非商业用途。参见 SEE ALSO
段
(译注:谁能想到,现在,2003 年,SCO会成为自由软件和开源软件界共同的敌人呢? 而Caldera 又在什么地方呢?)
作为对如此快速的商业化过程的补救 (As a counter-measure to the galopping commercialization,还请重新翻译), AT&T Bell Labs 贝尔实验室试图发起一个恢复性的项目 (?,a rescue project) ,使用他们的 Plan~9 操作系统。 它可以免费用于非商业用途,甚至包含了源代码。 但是它有一个专利许可证,that empedes the free development ( 还请解释词义)。 这种想法早已过时,因此 Plan~9 没有被主流开发者接受为自己的平台
真正的补救措施 (?, remedy) 是不断出现的自由操作系统 (386BSD, GNU/Linux, 等等.) 和 80 年代到 90 年代的自由软件工程。 他们实现了传统的 Unix 的特性和很多扩展,因此旧的 “XP体验” 不会丢掉 (such that the old experience is not lost)。 进入 21 世纪,类 Unix 系统重新成为计算机工业中的主导因素 --- 这要感谢自由软件
最重要的自由 roff 计划是 GNU 移植版本的 troff, 由 James Clark 建立,使用 它叫做 groff (GNU roff). 参见 groff(1) 中的概述
groff 系统仍然在继续开发。 它与传统 troff 兼容,但是还添加了很多扩展。 它是第一个可以在几乎所有操作系统上运行的 roff 系统并且 --- 它是自由开放的。 这使得 groff 成为现在 roff 的事实标准
一些 roff 实现提供了包装程序,使得人们可以简单地在命令行使用 roff 系统。 例如,GNU 的 roff 实现 groff(1) 提供了命令行选项来避免传统 troff 中过长的的命令管道; grog(1) 程序试着从文档猜测应当使用什么参数来运行 groff; 不习惯于指定命令行选项的人应当用 groffer(1) 程序来图形化地显示 groff 文件和手册页
预处理器产生 roff 代码,传给一个 roff 处理器 (例如,troff), 然后 roff 处理器接下来产生中间输出,传给一个后处理器程序, 用来打印或者产生最终输出
所有这些组件都使用它们自己的程序语言; 每种语言是与其他组件完全无关的。 此外,还可以包括为特殊目标而制作 (tailor) 的 roff 宏包
大多数 roff 文档中掺杂着使用一些包中的宏、 一个或多个预处理器的代码,还会添加 roff 语言中的一些元素。 用户很少需要用到 roff 排版语言的完整功能; 只有宏包的作者需要知道底层细节 (gory details)
有大量的自由/商业 roff 预处理器。 一些不能在所有系统上使用, 还有一些预处理器被认为是 roff 系统不可分割的部分。 传统的预处理器有
其他已知预处理器,但不是在所有系统上都可用,包括
roff 排版程序的输出以另外一种语言表示: intermediate output format 或 troff output. 这种语言最初详述在 [CSTR~#97]
中;它的 GNU 扩展记载在 groff_out(5) 中。 中间输出语言与高级的 roff 语言相比像一种汇编指令语言。 产生的中间输出是为一种特定的设备优化过的, 但是对于所有设备,这种语言都适用
roff 排版程序是整个 roff 系统的核心。 传统 roff 有两个排版程序: 对应字符设备的 nroff 和对应图形设备的 troff
通常, troff 这个名字泛指这两种排版程序 (is used as a general term to refer to both formatters)。
后处理器是将 troff 输出转化为一种适于某种特殊设备的格式的程序。 对于输出目标来说,roff 后处理器像是它们的设备驱动
每种设备都有为其优化的后处理器程序。. 后处理器解释中间输出,产生设备相关的代码,传送给设备
设备名和后处理器程序的名称是不固定的, 因为它们依赖于计算机的软硬件的能力。 例如, [CSTR~#54] 中提到的传统的设备名已经有了极大的改变。 旧的硬件不再存在,旧的图形转换程序与现代同等功能的程序相比太不精确了
例如,Postscript 设备 post 在传统 troff 中分辨率是 720,而 groff 的 ps 设备是 72000, 提高了 100 倍
现在,操作系统为大多数类似打印机的硬件提供了设备驱动, 因此不必为每个打印机写一个特殊的后处理器
文档中用到的宏包可以使用命令行选项
提供给排版程序, 参见 troff(1), 它们也可以在文档中指定,使用 roff 语言的 “包含文件” 命令,参见 groff(7).
著名的传统宏包有 man 用来处理传统手册页 mdoc 处理 BSD 样式的手册页;此类书籍、文档和信件的宏集合是 me (命名也许是根据它的创造者之名 Eric Allman 而来), ms (命名来自 Manuscript Macros/), 还有 mm (命名来自 Memorandum Macros/).
中. roff 语言是完整的编程语言,提供了命令 (request),宏定义,转义序列 (escape sequence), 字符串变量,数字或数量寄存器 (number or size registers),还有流程控制语句
Requests “命令” 是预定义的基础的排版命令,与 shell 提示下的命令类似。 用户可以定义类似 “命令” 的元素,使用 roff 的 “预定义” 元素。 用户定义的命令就被叫做 “宏” macros. 文档作者不会体会到 命令和宏 之间用法的任何区别; 它们都写在一行中,并以一个点 (dot, `.') 开始
Escape sequences “转义序列” 是以反斜杠 (backslash, ) 开始的 roff 元素。 它们可以被插入到任何地方,包括文本的一行中间。 它们用来实现不同的功能,可以使用
插入非 ASCII 字符,使用
改变字体,使用
插入行内注释,它们也可以转义特殊的控制字符,像这样 还有很多很多其他的功能
Strings “字符串” 是存储一个字符串的变量。 字符串以 .ds 命令存储。 存储的字符串可以用 \* 转义序列获得
Registers “寄存器” 存储数字和数量。 寄存器可以用 .nr 命令设置,然后用控制序列 \n
传统的宏包使用包名称作为文件扩展名,例如 file.me 意思是使用了 me 宏包的文件, file.mm 使用了宏包 mm, file.ms 用的是 ms, file.pic 则是 pic 等等
但是没有 roff 文档统一的命名方式,尽管 file.tr 用于 troff file 在现在和当时都是一样的 也许应当有 roff 文件扩展名的一个标准
文件扩展名与 less(1) 格式化工具结合,会非常好用。 它提供了用单一的方式打开各种输入的可能性, 方法是定义一个 shell 环境变量 LESSOPEN. 这种办法没有什么文档,这是一个例子:
lesspipe 可以是一个系统提供的命令或者是你自己的脚本。
当用 Emacs 编辑文档时可以输入 `M-x nroff-mode' 来改变模式,这里 M-x 意思是按住 Meta 键 (或 Alt) 同时点一下 x~
键
也可以在编辑器加载文件的时候,自动改变模式
Local Variables: mode: nroff End:
-*- nroff -*-
作为文件的第一行,也可以使 emacs 在加载文件时启用 nroff 模式。 不幸的是,一些程序例如 man 在这种方法中会出错;因此请不要用它
所有的 roff 排版程序都提供了自动的断行以及水平和竖直间距。 为了不干扰它的功能,应当注意以下几点:
下面的例子显示了最佳的 roff 编辑习惯是怎样的
这是一个 roff 文档的例子. 这是同一段中的下一句. 这是下一个句子,它比较长,分成了多 行;类似 `cf.' 这样的短语可以很容易地 识别,因为其中的“点”后面没有换行. (在输出时,它们仍会在同一段中.) (译注:如果使用中文的标点“。”就不用考虑这些, 但是每句话后面换行总是个好主意。少用 `.' 为妙)
除了 Emacs,其他一些编辑器也提供了 nroff 格式文件的支持,例如 vim(1), 它是 vi(1) 程序的扩展
(译注:为什么 2003 年 SCO 又会成为 M$ 的走狗呢?)
将 roff 作为一种“小语言”("little language") 来讲述的论文有
在 groff
中,手册页 groff(1) 包含了 groff 相关的所有文档的索引
其他系统中,你需要自己探索,但是 troff(1) 是个很好的起点
This document is distributed under the terms of the FDL (GNU Free Documentation License) version 1.1 or later. You should have received a copy of the FDL on your system, it is also available on-line at the
此文档是 groff, GNU roff 套件的一部分。 它的作者是 它的管理者是