1.简单了解预处理


    一个C程序的运行包括编译链接两个阶段,其实在编译之前预处理器首先要进行预处理操作,将处理完产生的一个新的源文件进行编译。可见预处理过程先于编译器对源代码进行处理. 在C 语言中,并没有任何内在的机制来完成如下一些功能:在编译时包含其他源文件、定义宏、根据条件决定编译时是否包含某些代码。要完成这些工作,就需要使用预处理程序。尽管在目前绝大多数编译器都包含了预处理程序,但通常认为它们是独立于编译器的。预处理过程读入源代码,检查包含预处理指令的语句和宏定义,并对源代码进行相应的转换。预处理过程还会删除程序中的注释和多余的空白字符。




预编译的主要作用


如下:


●将源文件中以”include”格式包含的文件复制到编译的源文件中。
●用实际值替换用“#define”定义的字符串。
●根据“#if”后面的条件决定需要编译的代码。




预处理指令是以#号开头的代码行。#号必须是该行除了任何空白字符外的第一个字符。#后是指令关键字,在关键字和#号之间允许存在任意个数的空白字符。整行语句构成了一条预处理指令,该指令将在编译器进行编译之前对源代码做某些转换。


部分预处理指令(仅作了解)


         指令              用途
         #                  空指令,无任何效果
         #include      包含一个源代码文件
         #define        定义宏
         #undef         取消已定义的宏
         #if               如果给定条件为真,则编译下面代码
         #ifdef          如果宏已经定义,则编译下面代码
         #ifndef        如果宏没有定义,则编译下面代码
         #elif            如果前面的#if给定条件不为真,当前条件为真,则编译下面代码
         #endif         结束一个#if……#else条件编译块
         #error         停止编译并显示错误信息




指令规则
●指令都是以#开始。#符号不需要在一行的行首,只要她之前有空白字符就行。在#后是指令名,接着是指令所需要的其他信息。
●在指令的符号之间可以插入任意数量的空格或横向制表符。
●指令总是第一个换行符处结束,除非明确地指明要继续。
●指令可以出现在程序中德任何地方。我们通常将#define和#include指令放在文件的开始,其他指令则放在后面,甚至在函数定义的中间。
●注释可以与指令放在同一行。


2.条件编译

顾名思义,条件编译指令将决定那些代码被编译,而哪些是不被编译的。可以根据表达式的值或者某个特定的宏是否被定义来确定编译条件。

条件编译是C语言提供的三种预处理功能的其中一种,这三种预处理包括:条件编译、文件包含、宏定义。

其中条件编译又有三种形式

第一种形式 (#if defined等价于#ifdef)


   #ifdef   标识符

       程序段1

   #else

       程序段2


   #endif

功能: 如果标识符已被 #define 命令定义过则对程序段1进行编译;否则对程序段2进行编译。

         如果没有程序段2(它为空),本格式中的#else可以没有,即可以写为:


   #ifdef  标识符

      程序段

    #endif


示例

#include <stdio.h>
#define WIN16 true
int main(void){
    #ifdef WIN16
        printf("The value of sizeof(int) is 2.\n");
    #else
        printf("The value of sizeof(int) is 4.\n");
    #endif
    return 0;
}


运行结果

The value of sizeof(int) is 2.

第4行插入了条件编译预处理命令,要根据 WIN16 是否被定义过来决定编译哪一个 printf 语句。而在程序的第2行已对 WIN16 作过宏定义,所以应对第一个 printf 语句进行编译。


程序第2行宏定义中,定义 WIN16 表示字符串 true,其实也可以为任何字符串,甚至不给出任何字符串,写为:

#define WIN16

也具有同样的意义。只有取消程序的第2行才会去编译第二个 printf 语句。

第二种形式 (#if !defined 等价于 #ifndef)


   #ifndef 标识符

      程序段1 

   #else 

      程序段2 

   #endif


与第一种形式的区别是将ifdef改为ifndef

功能: 如果标识符未被#define命令定义过则对程序段1进行编译,否则对程序段2进行编译。这与第一种形式的功能正相反。

第三种形式


   #if 常量表达式

      程序段1

   #else 

      程序段2

   #endif

功能:如常量表达式的值为真(非0),则对程序段1 进行编译,否则对程序段2进行编译。因此可以使程序在不同条件下,完成不同的功能。


请看下面的例子:

#import "ViewController.h"
#define  R    1
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    [self conditionalCompilation];
}
-(void)conditionalCompilation{
#if R
    NSLog(@"方法一");
#else
    NSLog(@"方法二");
#endif
}

运行结果


2016-04-01 15:48:47.085 测试[9686:219315]方法一


第2行宏定义中,定义R为1,因此在条件编译时,常量表达式的值为真,所以显示出button

最后

标识符: 在理论上来说可以是自由命名的,但每个头文件的这个标识符都应该是唯一的。标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线,如:stdio.h。

#ifndef _STDIO_H_

#define _STDIO_H_

/*程序段 */

#endif

条件编译的好处: 上面介绍的条件编译当然也可以用条件语句 if-else 来实现。 但是用条件语句将会对整个源程序进行编译,生成的目标代码程序较长,而采用条件编译,则根据条件只编译其中的程序段1或程序段2,生成的目标程序较短。如果条件选择的程序段很长,采用条件编译的方法是十分必要的。

3.其它一些标准指令:   

1>预定义的宏名
         ANSI C标准预定义了五个宏名,每个宏名的前后均有两个下画线,避免与程序员定义相同的宏名(一般都不会定义前后有两个下划线的宏)。这5个宏名如下:
●  __DATE__:   当前源程序的创建日期。
●    __FILE__:   当前源程序的文件名称(包括盘符和路径)。
●   __LINE__:   当前被编译代码的行号。
● __STDC__:    返回编译器是否位标准C,若其值为1表示符合标准C,否则不是标准C.
●  __TIME__:   当前源程序的创建时间。
 示例:     

#include<stdio.h>
int main(){
   int j;
   printf("日期:%s\n",__DATE__);
   printf("时间:%s\n",__TIME__};
   printf("文件名:%s\n",__FILE__);
   printf("这是第%d行代码\n",__LINE__);
   printf("本编译器%s标准C\n",(__STD__)?"符合":"不符合");
   return 0;
}

2>重置行号和文件名命令------------#line
        使用__LINE__预定义宏名赈灾编译的程序行号。使用#line命令可改变预定义宏__LINE__与__FILE__的内容,该命令的基本形如下:
        #line number[“filename”]
其中的数字为一个正整数,可选的文件名为有效文件标识符。行号为源代码中当前行号,文件名为源文件的名字。命令为#line主要用于调试以及其他特殊应用。
 示例:

#include<stdio.h> 
#line 1000
     int main(){
        printf("当前行号:%d\n",__LINE__);
        return 0;
     }

在以上程序中,在第4行中使用#line定义的行号为从1000开始(不包括#line这行)。所以第5行的编号将为1000,第6行为1001,第7行为1002,第8行为1003.
3>修改编译器设置命令 ------------#pragma
#pragma命令的作用是设定编译器的状态,或者指示编译器完全一些特定的动作。#pragma命令对每个编译器给出了一个方法,在保持与C语言完全兼容的情况下,给出主机或者操作系统专有的特征。其格式一般为:

#pragma Para

其中,Para为参数,可使用的参数很多,下面列出常用的参数:


Message参数

格式如:
#pragma message(消息文本)  当编译器遇到这条指令时,就在编译输出窗口中将消息文本显示出来。该参数能够在编译信息输出窗口中输出对应的信息,这对于源代码信息的控制是非常重要的,

code_seg参数

格式如:

#pragma code_seg([“section_name”[,section_class]])

它能够设置程序中函数代码存放的代码段,在开发驱动程序的时候就会使用到它。


once 参数 

格式如:
#pragma once
只要在头文件的最开始加入这条指令就能够保证头文件被编译一次。

4>产生错误信息命令 ------------#error

  #error命令强制编译器停止编译,并输出一个错误信息,主要用于程序调试。其使用如下:
  #error 信息错误
  注意,错误信息不用双括号括起来。当遇到#error命令时,错误信息将显示出来。
  例如,以下编译预处理器命令判断预定义宏__STDC__,如果其值不为1,则显示一个错误信息,提示程序员该编译器不支持ANSI C标准。

#if __STDC__!=1
      #error NOT ANSI C
    #endif