9.4 多源代码文件程序的编译

使用多个函数的最简单方法是将它们放在同一文件中,然后就像编译单个函数的文件一样对该文件进行编译。而其他方法就根据操作系统的不同而不同,以下几节将举例说明这些方法。

9.4.1 UNIX

首先假定UNIX系统下安装了标准的UNIX C编译器cc。文件file1.c和file2.c中包含有C函数。下面的命令将把这两个文件编译在一起并生成可执行文件a.out:

广告:个人专属 VPN,独立 IP,无限流量,多机房切换,还可以屏蔽广告和恶意软件,每月最低仅 5 美元

阅读 ‧ 电子书库

另外还将生成两个目标文件filel.o和file2.o。如果随后只更改了文件file1.c而file2.c没有改变,可以使用以下命令编译第一个文件并将其链接到第二个文件的目标代码:

阅读 ‧ 电子书库

在UNIX系统下有一个make命令可以自动管理多文件程序,本书不对此做深入讨论。

9.4.2 Linux

首先假定Linux系统下安装了GNU C编译器gcc。文件filel.c和file2.c中包含有C函数。下面的命令将把这两个文件编译在一起并生成可执行文件a.out:

阅读 ‧ 电子书库

另外还将生成两个目标文件file1.o和file2.o。如果随后只对file1.c进行了改动而file2.c不变,下面的命令可以对其进行编译并链接到file2.c的目标代码:

阅读 ‧ 电子书库

9.4.3 DOS命令行编译器

大多数DOS命令行编译器的工作机制同UNIX系统下的cc命令类似。一个不同之处在于DOS系统下目标文件的扩展名是.obj而不是.o。而且有些编译器并不生成目标代码文件,而是生成汇编语言或其他特殊代码的中间文件。

9.4.4 Windows和Macintosh编译器

Windows和Macintosh系统下的编译器是面向工程的。工程(project)描述了一个特定的程序所使用的资源。这些资源中包括源代码文件。使用这种编译器运行单文件程序时,必须创建工程。而对于多文件程序,需要使用相应的菜单命令将源代码文件加入到一个工程之中。而且,工程必须包括所有的源代码文件(扩展名为.c的文件)。但是,头文件(扩展名为.h的文件)不能包含在工程之中。因为工程只管理所使用的源代码文件,而使用哪些头文件需要由源代码文件中的#include指令确定。

9.4.5 头文件的使用

如果把main()函数放在第一个文件中而把自定义函数放在第二个文件中实现,那么第一个文件仍需要使用函数原型。如果把函数原型放在一个头文件中,就不必每次使用这些函数时输入其原型声明。这也正是C标准库的做法,比如把输入/输出函数的原型声明放在stdio.h中,把数学函数的原型声明放在math.h之中。对于包含自定义函数的文件也可以这样做。

编写程序的过程中需要经常使用C的预处理器定义常量。而定义的常量只能用于包含相应#define语句的文件。如果程序中的函数分别放在不同的文件之中,那么就必须使定义常量的#define指令对每个文件都可用。而直接在每个文件中键入该指令的方法既耗时又容易出错,同时也会带来一个维护上的问题:即如果修改了一个使用#define定义的数值,那么必须在每一文件中对其进行修改。比较好的解决方法是把所有的#define指令放在一个头文件中,然后在每个源代码文件中使用#include语句引用该头文件。

总之,把函数原型和常量定义放在一个头文件中是一个很好的编程习惯。我们考虑这样一个例子。假设需要管理4个连锁的旅馆。每一个旅馆都有不同的收费标准,但是对于一个特定的旅馆,其中的所有房间都符合同一种收费标准。对于预定住宿时间超过一天的人来说,第2天的收费是第一天的95%,而第3天的收费则是第2天的95%,等等(先不考虑这种策略的经济效益)。我们需要这样一个程序,即对于指定的旅馆和总的住宿天数可以计算出收费总额。同时程序中要实现一个菜单,从而允许用户反复进行数据输入直到选择退出。

程序清单9.9、程序清单9.10以及程序清单9.11列出了上述程序的源代码。第一个程序清单包含了main()函数,在main()函数中可以看出整个程序的组织结构。第二个程序清单包含所使用的函数,而且我们假设这些函数放在一个单独的文件中。最后,程序清单9.11列出了一个头文件,其中包含了程序的所有源文件使用的自定义常量和函数原型。前面讲过,在UNIX和DOS环境下,指令#include “hotels.h”中的双引号表示被包含的文件位于当前工作目录下(该目录一般包含源代码)。

程序清单9.9 usehotel.c控制模块

阅读 ‧ 电子书库

阅读 ‧ 电子书库

程序清单9.10 hotel.c函数支持模块

阅读 ‧ 电子书库

程序清单9.11 hotel.h头文件

阅读 ‧ 电子书库

阅读 ‧ 电子书库

下面是一个运行示例:

阅读 ‧ 电子书库

顺便提一下,程序中有几处很具特色。比如函数menu()和getnights()通过检测scanf()的返回值来跳过输入的非数字数据,并且调用函数scanf(“%*s”)来跳至下一空白字符。请注意menu()中的以下代码如何检查出非数字的输入和超出范围的数据:

阅读 ‧ 电子书库

这段代码运用了C的两个运算规则:逻辑表达式从左向右运算;并且一旦结果明显为假,运算会立刻停止。在本例中,只有确定scanf()已成功地读取了一个整型数值后,变量code的数值才会被检查。

用函数分别实现各个独立的功能需要使用这种精练的语句。当然第一次编写menu()和getnights()时可能只使用了简单的scanf()函数而没有数据检查功能。然后,就可以根据基本程序的运行情况对每个模块进行改进。