一、预处理的由来: 
     在C++的历史发展中,有很多的语言特征(特别是语言的晦涩之处)来自于C语言,预处理就是其中的一个。C++从C语言那里把C语言预处理器继承过来(C语言预处理器,被Bjarne博士简称为Cpp,不知道是不是C Program Preprocessor的简称)。
二、常见的预处理功能: 
     预处理器的主要作用就是:    把通过预处理的内建功能对一个资源进行等价替换,最常见的预处理有: 文件包含,条件编译、布局控制和宏替换4种。 
     文件包含:    #include 是一种最为常见的预处理,主要是做为文件的引用组合源程序正文。 
     条件编译:    #if,#ifndef,#ifdef,#endif,#undef等也是比较常见的预处理,主要是进行编译时进行有选择的挑选,注释掉一些指定的代码,以达到版本控制、防止对文件重复包含的功能。 
     布局控制:    #progma,这也是我们应用预处理的一个重要方面,主要功能是为编译程序提供非常规的控制流信息。 
     宏替换:    #define,这是最常见的用法,它可以定义符号常量、函数功能、重新命名、字符串的拼接等各种功能。 
     
三、预处理指令: 
     预处理指令的格式如下: 
     # define tokens 
     #符号应该是这一行的第一个非空字符,一般我们把它放在起始位置。如果指令一行放不下,可以通过反斜杠“/”进行控制,例如: 
     #define Error /
                 if(error) exit(1)   
等价于 
     #define Error if(error) exit(1) 
     不过我们为了美化起见,一般都不怎么这么用,更常见的方式如下: 
     # ifdef __BORLANDC__ 
             if_true<(is_convertible<value,named_template_param_base>::value)>:: 
             template then<make_named_arg, make_key_value>::type Make; 
     # else 
             enum { is_named = is_named_parameter<value>::value }; 
             typedef typename if_true<(is_named)>::template 
             then<make_named_arg, make_key_value>::type Make; 
     # endif 
******************************************************************* 
下面我们看一下常见的预处理指令: 
     #define         宏定义 
     #undef          取消宏 
     #include        文本包含 
     #ifdef            如果宏被定义就进行编译 
     #ifndef          如果宏未被定义就进行编译 
     #endif           结束编译块的控制 
     #if                表达式非零就对代码进行编译 
     #else            作为其他预处理的剩余选项进行编译 
     #elif              这是一种#else和#if的组合选项 
     #line             改变当前的行数和文件名称 
     #error            输出一个错误信息 
     #pragma        为编译程序提供非常规的控制流信息 
******************************************************************* 
     下面我们对这些预处理进行一一的说明,考虑到宏的重要性和繁琐性,我们把它放到最后讲。
四、文件包含指令: 
     这种预处理使用方式是最为常见的,平时我们编写程序都会用到,最常见的用法是: 
     #include <iostream>             file://标准库头文件 
     #include <iostream.h>          file://旧式的标准库头文件 
     #include "IO.h"                     file://用户自定义的头文件 
     #include "../file.h"                 file://UNIX下的父目录下的头文件 
     #include "/usr/local/file.h"      file://UNIX下的完整路径 
     #include "..//file.h"                file://Dos下的父目录下的头文件 
     #include "//usr//local//file.h"   file://Dos下的完整路径
     这里面有2个地方要注意: 
     1、我们用<iostream>还是<iostream.h>? 
        我们主张使用<iostream>,而不是<iostream.h>,为什么呢?我想你可能还记得我曾经给出过几点理由,这里我大致的说一下: 
        首先,.h格式的头文件早在98年9月份就被标准委员会抛弃了,我们应该紧跟标准,以适合时代的发展。 
        其次,iostream.h只支持窄字符集,iostream则支持窄/宽字符集。 
        还有,标准对iostream作了很多的改动,接口和实现都有了变化。 
        最后,iostream组件全部放入namespace std中,防止了名字污染。 
     2、<io.h>和"io.h"的区别? 
        其实他们唯一的区别就是搜索路径不同: 
        对于#include   <io.h> ,编译器从标准库路径开始搜索 
        对于#include   "io.h" ,编译器从用户的工作路径开始搜索
五、编译控制指令: 
     这些指令的主要目的是进行编译时进行有选择的挑选,注释掉一些指定的代码,以达到版本控制、防止对文件重复包含的功能。 
     使用格式,如下: 
     1、 
       #ifdef   identifier 
               your code 
       #endif 
       如果identifier为一个定义了的符号,your code就会被编译,否则剔除 
     2、 
       #ifndef identifier 
               your code 
       #endif 
       如果identifier为一个未定义的符号,your code就会被编译,否则剔除 
     3、 
       #if   expression 
            your code 
       #endif 
       如果expression非零,your code就会被编译,否则剔除 
     4、 
       #ifdef identifier 
              your code1 
       #else 
              your code2 
       #endif 
       如果identifier为一个定义了的符号,your code1就会被编译,否则your code2就会被编译 
     5、 
       #if    expressin1 
             your code1 
       #elif expression2 
             your code2 
       #else 
             your code3 
       #enif   
       如果epression1非零,就编译your code1,否则,如果expression2非零,就编译your code2,否则,就编译your code3
其他预编译指令 
     除了上面我们说的集中常用的编译指令,还有3种不太常见的编译指令:#line、#error、#pragma,我们接下来就简单的谈一下。 
     #line的语法如下: 
       #line number filename 
     例如:#line 30   a.h      其中,文件名a.h可以省略不写。 
     这条指令可以改变当前的行号和文件名,例如上面的这条预处理指令就可以改变当前的行号为30,文件名是a.h。初看起来似乎没有什么用,不过,他还是有点用的,那就是用在编译器的编写中,我们知道编译器对C++源码编译过程中会产生一些中间文件,通过这条指令,可以保证文件名是固定的,不会被这些中间文件代替,有利于进行分析。 
     #error语法如下: 
         #error   info 
     例如:
            #ifndef UNIX 
              #error This software requires the UNIX OS. 
            #endif 
     这条指令主要是给出错误信息,上面的这个例子就是,如果没有在UNIX环境下,就会输出This software requires the UNIX OS.然后诱发编译器终止。所以总的来说,这条指令的目的就是在程序崩溃之前能够给出一定的信息。 
     至于#pragma,我们在《解析#pragma指令 》一文中有过介绍,我们在这里再补充几句,#pragma是非统一的,他要依靠各个编译器生产者,例如,在SUN C++编译器中: 
          // 把name和val的起始地址调整为8个字节的倍数 
          #progma align 8 (name, val) 
          char    name[9]; 
          double val; 
         file://在程序执行开始,调用函数MyFunction 
         #progma init (MyFunction)
预定义标识符 
     为了处理一些有用的信息,预处理定义了一些预处理标识符,虽然各种编译器的预处理标识符不尽相同,但是他们都会处理下面的4种: 
     __FILE__   正在编译的文件的名字 
     __LINE__   正在编译的文件的行号 
     __DATE__   编译时刻的日期字符串,例如: "25 Dec 2000" 
     __TIME__   编译时刻的时间字符串,例如: "12:30:55" 
     例如:cout<<"The file is :"<<__FILE__"<<"! The lines is:"<<__LINE__<<endl;
预处理何去何从 
     在《浅析C++里面的宏》一文中,我们提到了如何取代#include预处理指令,我们在这里就不再一一讨论了。 
     C++并没有为#include提供替代形式,但是namespace提供了一种作用域机制,它能以某种方式支持组合,利用它可以改善#include的行为方式,但是我们还是无法取代#include。 
     #progma应该算是一个可有可无的预处理指令,按照C++之父Bjarne的话说,就是:“#progma被过分的经常的用于将语言语义的变形隐藏到编译系统里,或者被用于提供带有特殊语义和笨拙语法的语言扩充。” 
     对于#ifdef,我们仍然束手无策,就算是我们利用if语句和常量表达式,仍然不足以替代它,因为一个if语句的正文必须在语法上正确,满足类检查,即使他处在一个绝不会被执行的分支里面。 
     最后,我们以Bjarne博士的话作为结尾:“最后,在许多年之后,将Cpp放逐刀程序开发环境里,与其他附加性语言工具放到一起,那里才是她应该呆的地方。”
转自:http://www.cnblogs.com/lidabo/archive/2012/08/27/2658914.html

C++中的预处理的更多相关文章

  1. C#中的预处理指令

    C#中的预处理指令 作为预处理中的一对:#region name ,#endregion可能是大家使用得最多的,我也常用它来进行代码分块,在一个比较长的cs文件中,这么做确实是一件可以让你使代码更清晰 ...

  2. C#中的预处理指令详解

    这篇文章主要介绍了C#中的预处理指令详解,本文讲解了#define 和 #undef.#if.#elif.#else和#endif.#warning和#error.#region和#endregion ...

  3. PDO中的预处理

    PDO中的基本的原理和步骤和MySQL中的预处理都是一样的,只不过就是把MySQL中的预处理所有命令行的语法封装成了PDO对象的几个公开的方法而已! 1.发送预处理语句 此时,我们需要调用pdo对象的 ...

  4. VS中添加预处理宏的方法

    VS中添加预处理宏的方法 除了在.c及.h中添加宏定义之外,还可以采用如下方法添加宏定义: 1.若只需要定义一个宏(如#define DEBUG),可以右键点击工程-->属性-->c/c+ ...

  5. C语言中的预处理命令

    预处理功能是C语言的重要功能. 问:为什么要预处理,什么是预处理? 答:我们知道高级语言的运行过程是通过编译程序(编译器)把源代码翻译成机器语言,实现运行的.编译程序的工作包含:语法分析.词法分析.代 ...

  6. C 语言中的预处理

    C 语言中以 # 开头的就是预处理指令,例如 #include . 预处理指令的用途 所有的预处理指令都会在 GCC 编译过程的预处理步骤解析执行,替换为对应的内容.在下一步编译过程中,看不到任何预处 ...

  7. Keil 中的预处理命令const

    在keil中,宏定义是一个重要内容.无参数的宏作为常量,而带参数的宏则可以提供比函数更高的调用效率.但预处理只是进行简单的文本代替,而不做语法检查,所以会存在一些问题. 例如:#define BUFS ...

  8. 机器学习中数据清洗&预处理

    数据预处理是建立机器学习模型的第一步,对最终结果有决定性的作用:如果你的数据集没有完成数据清洗和预处理,那么你的模型很可能也不会有效 第一步,导入数据 进行学习的第一步,我们需要将数据导入程序以进行下 ...

  9. pytorch中检测分割模型中图像预处理探究

    Object Detection and Classification using R-CNNs 目标检测:数据增强(Numpy+Pytorch) - 主要探究检测分割模型数据增强操作有哪些? - 检 ...

随机推荐

  1. JPush极光推送Java服务器端API

    // 对android和ios设备发送 JPushClient jpush = new JPushClient(masterSecret, appKey);   // 对android和ios设备发送 ...

  2. 黑马程序员——JAVA基础之编码表

    ------- android培训.java培训.期待与您交流! --------- 字符编码  字符流的出现为了方便操作字符.  更重要是的加入了编码转换.  通过子类转换流来完成. •  I ...

  3. ps磨皮

    光滑磨皮步骤: 1.用高斯模糊滤镜模糊皮肤,用蒙版控制范围,去掉较为明显的杂色及瑕疵.可以高斯模糊重复多次,去掉明显的杂色. 2.用涂抹工具处理细小的瑕疵及加强五官等部位的轮廓: 3.整体美白及润色 ...

  4. [转]UDP穿透NAT的原理与实现(UDP“打洞”原理)

    NAT(The IP Network Address Translator) 的概念和意义是什么? NAT, 中文翻译为网络地址转换.具体的详细信息可以访问RFC 1631 - http://www. ...

  5. Vmware vsphere webservice sdk 连接打开慢的问题

    还在为VimService实例化速度慢的问题烦恼吗?这有一篇文章可以帮你解决问题,英文水平所限,就不翻译了,原文地址http://kb.vmware.com/selfservice/microsite ...

  6. XML与JSON的对比

    XML与JSON的对比 1.各自定义 XML 扩展标记语言 (Extensible Markup Language, XML) ,用于标记电子文件使其具有结构性的标记语言,可以用来标记数据.定义数据类 ...

  7. ssh远程连接错误

    在平时工作中,有时候需要SSH登陆到别的Linux主机上去,但有时候SSH登陆会被禁止,并弹出如下类似提示: @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ ...

  8. yii2 使用twig 模板引擎

    yii2 默认使用PHP 和html 混合的方式来写视图层,但我个人还是喜欢纯模板语言的方式.而且已经非常习惯使用twig的语法,最近想使用yii2进行开发,所以还是选择使用twig视图引擎. git ...

  9. C#:org.in2bits.MyXls 文本格式日期 转换,以及设置单元格格式,保留两位小数点

    org.in2bits.MyXls  Excel导入日期格式的处理 表格内容为 2014-7-22 ,导入后显示为 41842 等于一个数值,根本不是日期,后来百度了一下,发现要做如下处理: stri ...

  10. EXCEL datatable 根据列名自动写入到相应属性、字段或列中

    string path = openFileDialog1.FileName; try { DataTable dt = ExcelHelper.ExcelInput(path); ; ; ; ; i ...