现在的位置: 首页 > 综合 > 正文

C语言预处理、结构体和make的使用–C语言预处理命令

2013年09月11日 ⁄ 综合 ⁄ 共 3571字 ⁄ 字号 评论关闭

  在C语言程序中加入一些编译预处理命令可以提高编程效率,回忆编译速度。预处理命令是在编译源程序前先对源程序进行处理,例如,在程序中使用“#defineMAX 256“这条命令定义一个符号常量MAX,则在预处理时将程序中出现的所有MAX替换为256。预处理完成后,编译器(如gcc)开始编译源程序以生成可执行代码。需要注意的是,预处理命令并不是C语言的一部分,因此每条编译预处理命令不需要以分号来结束。

    一、宏定义

    C语言标准允许在程序中用一个标识符来表示一个字符串,称为宏。标识符称为宏名。在编译预处理时,将程序中所有的宏名用相应的字符串来替换,这个过程称为宏替换。宏分为两种:无参数的宏和有参数的宏。

    1、无参数宏

    无参数宏定义的一般形式为:

#define 标识符 字符串

    “#”代表本行是编译预处理命令。define是宏定义的关键词,标识符是宏名。字符串是宏名所代替的内容,可以是常数、表达式等。

     注意:宏定义和其他编译预处理命令不是以分号结尾的。

    例5-1 下面是一个使用无参数宏的程序

#include <stdio.h>

#define PI 3.1415926

 

int main()

{

int r = 100;

double length = 2*PI*r;

printf("The circumference is %f\n",length);

return 0;

}

程序说明:(1)本程序使用PI来代表3.1415926。宏替换是在程序中用相应的字符串来替换害名,编译器预处理程序对它不作任何检查。如果有错误,只能在编译程序时才会被编译器发现。(2)习惯上,宏名都用大写字母。当然也可以用小写字母。(3)使用宏代替一个字符串,可以减少程序中重复书写某些字符串的工作量。可以用一个有意义的宏句来代表无规律的字符串,提高程序的可读性,同时修改起来也方便。如果要把程序中的PI值改为3.14,则只要修改#define这一行即可。如果没有使用宏,那么就要查找程序并修改所有的PI值。(4)宏的作用范围是从宏定义开始到本源程序文件结束为止。也可以使用#undef来提前终止作用范围。例如:

#define MAX 256

int main()

{

    ...

}

#undef MAX

int f()

{

    ...

}

    由于使用了#undef,使宏名MAX只在main函数中有效。(5)宏定义允许嵌套。例如:

#define MIN 128

#define MAX MIN*2

定义MAX宏时使用了前面已经定义的MIN。

    2、有参数宏

    有参数宏的宏类似于有参数的函数,其定义的一般形式为:

#define 标识符(形参表) 字符串

    如果有多个形参,像函数参数一样以逗号隔开。在程序中使用有参数宏的形式是:

标识符(实参表)

    例5-2演示了有参数宏的实现方法。

#include <stdio.h>

#define MAX(x,y) (x>y?x:y)

 

int main()

{

int a = 5,b = 10,max;

 

max = MAX(5,10);

printf("The max between(%d,%d) is %d\n",a,b,max);

return 0;

}

程序说明:经过编译预处理max = MAX(a,b)就替换max=(a>b?a:b)。

    程序第二行的宏定义中表达式x>y?x:y两边的括号不是必需的,但出于良好的编程规范应该加上。如果没有括号往往会导致一些意想不到的问题。比如有一个宏定义:

#defineMUL(x,y) x*y

    在程序中有:

int a=5,b=10,c;

c=MUL(a+1,b+1);

    那么进行宏替换,a+1是x的实参,b+1是y的实参,替换后的结果为:

c=a+1*b+1;

    显然这是不符合要求的。应该按照如下方式进行宏定义:

#define MUL(x,y) (x)*(y)

    此时宏展开后:

c = (a+1)*(b+1);

    定义有参数的宏时,应该注意:

    宏名与形参表的圆括号之间不能有空格,否则会导致错误。例如:#defineMUL(x,y) (x)*(y),MUL与“(”之间不能有空格。

    宏定义中,字符串内的形式参数最好用括号括起来,以避免错误。例如上面的形参都用括号括起来。

    带参数的宏与函数的比较:

    有参数宏的形式参数不是变量,不分配内存空间,无需说明数据类型。而函数的形参是变量,要分配内存空间,在函数定义时要指明参数的数据类型。

    预处理程序认为有参数宏的是字符串,并用它去替换形参。如上面的例子中,用a+1去替换x,而不是先计算a+1的值再去替换x。如果是函数,则先计算a+1的值,再把这个值传递给x。

    使用宏的次数较多时,宏替换后源程序一般会变长。而函数调用不会使程序变长。宏替换不会占用运行时间,只是编译的时间稍微变长一点。而函数调用则会占用运行时间。一般用宏来代表一些较为简单的表达式比较合适。

    二、文件包含

    文件包含预处理命令#include前面已经使用过了。它把指定源文件的全部内容包括到当前源程序文件中,其一般形式为:

#include <文件名>

或者

#include "文件名”

    文件包含命令是把指定文件的全部内容包括进来,插入到命令所在位置,取代原来的命令行。由当前源文件和指定文件组成一个文件,一起进行编译。

    一个#include只能包含一个文件,要包含多个文件,需要使用多个#include命令。例如:

#include <stdio.h>

#include <string.h>

#include <stdlib.h>

    一个大的程序往往被分成多个模块,由多个程序员分别编写。公用的信息,如常量定义、函数声明,可以单独放在一个文件中,在其他文件的开头使用#include命令包含进来。这样可以避免每个文件头部都去书写那些公用量,既节省了时间又可以防止出错的可能。

    文件包含命令中的文件名既可以用尖括号也可以用双引号括起来,它们的区别在于查找指定文件的位置不同。尖括号只在缺省目录里找指定文件,缺省目录是由用户设置的编程环境决定的。双引号则先在源程序文件所在的当前目录里查找指定文件,如果没有找到再到缺省目录里找。如果指定文件与当前编写中的源程序处在同一个目录里,就必须使用双引号来包含该文件,否则编译程序时编译器会报告找不到指定的头文件。

    三、条件编译

    一般情况下,源程序中所有的行都被编译。有时希望其中一部分内容只在某个条件成立或不成立时才去编译,也就是对一部分内容指定编译的条件,这就是条件编译。

    条件编译使用范式:

   1、范式一

#ifndef 标识符

程序段1

#endif

其含义是:如果没有定义标识符,就编译程序段1。这里的程序段1既可以是语句组,也可以是命令行。使用示例:

#ifndef _getkey_h

#define _getkey_h

#include<sys/types.h>

#endif

这段代码的含义是:如果没有定义符号常量_getkey_h,就定义该常量并且包含头文件sys/types.h。

2、范式2

#ifndef 标识符

程序段1

#else

程序段2

#endif

其含义是:如果没有定义标识符,就编译程序段1,否则编译程序段2。

3、范式3

#ifdef 标识符

程序段1

#endif

其含义是:如果定义了标识符,就编译程序段1,否则不编译该程序段。

使用示例:

#ifdef DEBUG

printf("a=%d,b=%d",a,b);

#endif

在调试程序时,可以在源程序头部加入如下语句:

#define DEBUG

这样在软件开发阶段,编译运行程序时会输出变量a,b的值。当程序调试完毕,在源程序文件头部删除这一行,则用户运行时不会输出a,b的值。这里打印出a,b值只是供调试使用。

4、范式四

#ifdef 标识符

程序段1

#else

程序段2

#endif

其含义是:如果定义了标识符,就编译程序段1,否则编译程序段2。

    5、范式五

#if 表达式

程序段1

#endif

其含义是:如果表达式成立,就编译程序段1,否则不编译该程序段。

使用示例:

#include <stdio.h>

#define MAX(x,y) (x>y?x:y)

...

int a = 5,b = 10,c;

...

#if c

c = MAX(a,b);

#endif

如果变量c存在,就调用宏MAX(a,b)获得a,b的最大值,并把该值赋给变量c。

    6、范式六

#if 表达式

程序段1

#else

程序段2

#endif

其含义是:如果表达式成立,就编译程序段1,否则编译程序段2。

    事实上,不用条件编译而直接用if...else语句也可以达到要求。但采用条件编译,可以减少被编译的语句,从而减少可执行程序的长度,缩短程序运行时间。当条件编译的程序段比较多时,可执行程序的长度可以大大减少。

抱歉!评论已关闭.