快捷搜索:

C语言的那些小秘密之预处理

C语言的那些小秘密之预处理

  预处理C语言的一个重要知识点,它能改善程序设计的环境,有助于编写易移植、易调试的程序。因此,我们有必要掌握好预处理命令,在自己编程的时候灵活的使用它,使得编写的程序结构优良,更加易于调试和阅读。接下来我尽可能的把预处理中重要知识点向读者讲解清楚,使读者能够在自己以后编程的过程中熟练的使用预处理命令。

本文引用地址:

  C语言的预处理主要有三个方面:

  1、文件的包含

  2、宏定义

  3、条件编译

  一、文件包含的形式有下面两种

  1、#include "文件名"

  2、#include <文件名>

  它们之间的区别在于:<文件名>系统到头文件目录查找文件, "文件名"则先在当前目录查找,如果没有才到头文件目录查找;当然我们也可以使用在命令行来指定头文件路径方法。还要注意就是如果在源文件包含的头文件之间出现调用的情况,那么被调用的头文件要出现在调用头文件的前面。

  二、宏定义

  宏定义的使用有两种形式,一种不带参数,而另外一种带参数。

  1、不带参数

  格式: #define 标识符 字符串

  相信上面这个格式大家并不陌生,下面还是来看看如何使用吧。当然在讲解之前我们的看看使用过程中的如下几个注意要点:

  (1)预处理不做语法检查,所以我们选用的时候要尤其小心

  (2)宏定义写在函数的花括号外边,作用域为其后的程序,通常在文件开头部分,直到用#undef命令终止宏定义的作用域

  (3)不要在字符串中使用宏,如果宏名出现在字符串中那么将按照字符串进行处理

  下面来看段代码的使用。

  [html] view plaincopy#include

  #define N 9

  int main ()

  {

  int i,a[N];

  for(i=0;i

  {

  a[i]=i;

  printf("%d\t",a[i]);

  if((i+1)%3==0)

  printf("\n");

  }

  //#undef N

  printf("%d\n",N);

  }

  运行结果为:

  [html] view plaincopy0 1 2

  3 4 5

  6 7 8

  9

  Press any key to continue

  我们在此主要是介绍下宏的作用域问题,当在以上代码中注释掉#undef N时,接下来的打印语句能够正常的打印出;但是当我们没有注释掉#undef N的时候就会出现error C2065: 'N' : undeclared identifier错误,提示N没有定义。接下来看看带参数的宏的使用。

  2、带参数

  #define 宏名(参数表) 字符串

  注意要点:

  (1)宏名和参数的括号间不能有空格

  (2)宏替换只作替换,不做计算,不做表达式求解,这点要尤其注意

  (3)函数调用在编译后程序运行时进行,并且分配内存。宏替换在编译前进行,不分配内存

  (4)宏的哑实结合(所谓的哑实结合类似于函数调用过程中实参替代形参的过程)不存在类型,也没有类型转换。

  (5)宏展开使源程序变长,函数调用不会

  下面来看看linux下一个典型的应用:

  #define min(x,y) ({ typeof(x) _x = (x); typeof(y) _y = (y); (void) (&_x == &_y); _x < _y ? _x : _y; })

  #define max(x,y) ({ typeof(x) _x = (x); typeof(y) _y = (y); (void) (&_x == &_y); _x > _y ? _x : _y; })

  在上面的两个宏中我们发现有这么一句代码(void) (&_x == &_y);可能不少读者有点发懵的感觉,这啥意思呢?!其实我们细细分析就知道,首先看看“==”,这是一个逻辑表达式,它要求两边的比较类型必须一致,如果我们的&x和&y类型不一致,如一个为char*,另一个为int*,不是同一个类型,当我们使用gcc编译的时候就会出现警告信息,vc6则会报错error C2446: '==' : no conversion from 'char *' to 'int *'。这句代码(void) (&_x == &_y); 在此的功能就相当于执行一个简单的判断操作,我们用来判断x和y的类型是否一致。别小看了这句代码,如果学会了使用它会给你的代码带来不少的便捷。下面给出一个小小的事例:

  [cpp] view plaincopy#include

  void print()

  {

  printf("hello world!!!\n");

  return ;

  }

  void main(int argc,char*argv)

  {

  print();

  return ;

  }

  运行结果为:

  

  [cpp] view plaincopyhello world!!!

  Press any key to continue

  现在我们来修改下代码后看看运行结果:

  [cpp] view plaincopy#include

  void print()

  {

  printf("hello world!!!\n");

  return ;

  }

  void main(int argc,char*argv)

  {

  #define print() ((void)(3))

  print();

  return ;

  }

  运行结果为:

  [cpp] view plaincopyPress any key to continue

您可能还会对下面的文章感兴趣: