博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
预处理指令
阅读量:6689 次
发布时间:2019-06-25

本文共 2376 字,大约阅读时间需要 7 分钟。

预处理指令

C源码在进行编译前会先经过预处理,预处理指令均以#开头,结尾没有分号(;)不是C语句。预处理器采用语言符号的分析方式,将空格作为区分标志之间的符号,一行作为一条指令,“\”可以将预处理指令延伸到下一行。

一、宏定义

1.不带参数的宏定义

  (1)不带参数的宏定义就是用一个标识符(宏名)来代表一个字符串。它的一般形式为

   #define Macro Str 

  在预处理的时候程序中的宏名Macro被替换为字符串Str,这个过程称为宏展开。 

  (2)#define指令出现在程序中函数的外面,宏名的有效范围为该指令行起到本源文件结束或#undef

  (3)宏展开只是简单的字符串替换,简单宏常用于定义常量,宏没有类型,也没有优先级的概念,使用定义常量主要用于指定数组长度 #define ayyLength 256 ,建议尽量使用constenum代替宏定义常量 const int arrLen 256; 。建议不使用宏定义类型 #define Status int 而是用typedef关键字 typedef Status int; 

  (4)宏不是C语句不能加分号,否则会将分号一起代入。考虑到优先级问题,其中表达式也可能需要括号。

  (5)宏代替的字符串可以是常量也可以表达式,格式描述符,语句(甚至是它们的一部分)等任何C程序中出现的字符串。 当然,宏代替的字符串中也可以包含已定义的宏名。 

   #undef macro 

   #undef指令可以终止宏名的定义。

2.带参数的宏定义

  (1)

  

#define Macro(argus) str

 

  对带参数的宏展开 是将带实参的宏 按照#define指令行中按从左到右的顺序进行置换。宏名和带参数的括号之间不能加空格,否则将成为无参数宏,空格后的每一个字符都将作为替代字符串的一部分。

  (2)带参数的宏不是函数,它只是进行简单字符代换 。定义宏时参数和字符串可以是任意的,但在定义时要注意标识符不能出现重名【#define Macro(Macro) str】(利用代码块作用域)在实际调用时参数可能是单个数据对象也可能是表达式,由于宏只进行简单文本代换考虑到优先级和结合性的问题,建议在定义宏时将参数用圆括号括起以作为一个独立单位。

  (3)含参宏类似于inline函数,其调用也是采用传址的方式(真正的在调用点嵌入函数代码)。宏无类型,其参数也没有类型,只是一个符号代表,含参宏可以作为模板函数(C++引入,不同类型的参数可以使用同一段函数体)。

  (4)宏与函数相比没有参数传递和返回值的限制使用更加自由灵活,而函数相对独立便于完成较复杂的任务。函数调用需较多时间处理内存等,而宏不需要。

3.宏定义中的运算符:

  (1)对程序作预处理前,编译器会进行翻译处理。编译器首先把源码中的字符映射到源字符集。然后编译器查找反斜线后紧跟换行符(这里指按下回车键在源码中产生的换行符而非转义字符'\n')的实例并删除这些实例。所以反斜线加回车键可以将宏定义扩展到多行。

  (2)#运算符与参数结合可以那参数名转换为相应字符串。例如:x是宏的形参,实参为1#x将被替换为"1"(字符串)x将被替换为1(数值常量)

  (3)##运算符可以把两个语言符号组合成一个语言符号。例如:n是宏的形参,实参为1时,x##n将变为标识符x1,如果无##编译器将把xn当做一个语言符号,在宏参数中中无法找到于是不进行替换。

二、文件包含处理

   #include<filename> 或  #include"filename" 

  • 尖括号中的文件名优先在编译器安装目录中查找(通常是标准库),双引号中的文件优先在工作目录中查找(自定义库)
  • 文件包含处理是指将另一个源文件的全部内容包含进来,即将另外的文件内容包含到本文件之中,插入到当前位置,代替预处理指令然后进行编译得到一个目标文件。
  • 头文件中只有函数声明和宏定义,真正的实现在库中。在链接(linking)时,库才被链接进来。
  • 这种常用在文件头部的被包含文件被称为头文件(header),常以.h作为后缀。当然不用.h作为后缀用.c作为后缀也是可以的,但是用.h更能表示此文件的性质。
  • 同样,#include指令不一定要出现在文件首部。应当注意,被包含文件修改后,凡包含此文件的所有文件都要全部重新编译。文件包含处理是将要包含的文件的内容代替预处理指令,成为源文件的一部分。

三、条件编译

  条件编译使得程序中的一部分内容只在满足一定条件时才进行编译或不进行编译。

1.ifdef指令

 

#ifdef Label     Block1#else     Block2#endif

 

  若指定的标识符已经被#define指令定义过,则对程序段1进行编译否则对程序段2进行编译。可以用于程序调试等。

  也可以用

  

#ifdef      Block#endif

 

 

  2.ifndef指令

 

#ifndef Label     Block1#else    Block2#endif

 

   与1正好相反,若标识符未被定义则编译程序段1,否则编译程序段2

   为了避免头文件重复包含,通常使用条件编译:

  

#ifndef STDIO_H#define STDIO_H     ...#endif

 

 

  3.if指令

 

#if expr     Block1#else    Block2#endif

 

  表达式为真时编译程序段1,否则编译程序段2。

  不用条件编译指令而用if语句同样也可以实现,但是那样做目标程序长(所有语句都参加编译),运行时间长(在if语句处需进行逻辑判断)。

 

转载地址:http://gkkoo.baihongyu.com/

你可能感兴趣的文章
js中用gb2312编码解码
查看>>
细谈测试---我的启示录
查看>>
Unity自定义mesh绘制
查看>>
nagios安装
查看>>
好用的省、市、地区联动JS封装类
查看>>
Dalvik字节码的类型,方法与字段表示方法
查看>>
(新方法)图片一句话***(免杀)
查看>>
Warning:Even Exadata has a wrong memlock setting
查看>>
java异常处理
查看>>
懒人笔记:mysql基础4
查看>>
用集算器实现报表的外置存储过程数据源效果
查看>>
ProtocolBuffer Mac安装步骤(亲测可用)
查看>>
javascrip对表格的操作(二)
查看>>
构建需求响应式亿级商品详情页
查看>>
CentOS 7 下安装MySQL WorkBench
查看>>
Nginx 教程- 获取真实IP模块 - http_realip_module
查看>>
SQL语句教程(04) AND OR
查看>>
Python 中有关中文编码解码小记
查看>>
EBS 12.1.3 db 11.2.3 dg AND DG SWITCH OVER
查看>>
Oracle中的JOIN
查看>>