`
javasogo
  • 浏览: 1825892 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

C++中的预处理

 
阅读更多

关 键 词:C

阅读提示:在C++的历史发展中,有很多的语言特征(特别是语言的晦涩之处)来自于C语言,预处理就是其中的一个。C++从C语言那里把C语言预处理器继承过来(C语言预处理器,被Bjarne博士简称为Cpp,不知道是不是C Program Preprocessor的简称)。

一、预处理的由来:在C++的历史发展中,有很多的语言特征(特别是语言的晦涩之处)来自于C语言,预处理就是其中的一个。C++从C语言那里把C语言预处理器继承过来(C语言预处理器,被Bjarne博士简称为Cpp,不知道是不是C Program Preprocessor的简称)。二、常见的预处理功能:预处理器的主要作用就是把通过预处理的内建功能对一个资源进行等价替换,最常见的预处理有:文件包含,条件编译、布局控制和宏替换4种。文件包含:#include 是一种最为常见的预处理,主要是做为文件的引用组合源程序正文。条件编译:#if,#ifndef,#ifdef,#endif,#undef等也是比较常见的预处理,主要是进行编译时进行有选择的挑选,注释掉一些指定的代码,以达到版本控制、防止对文件重复包含的功能。布局控制:#progma,这也是我们应用预处理的一个重要方面,主要功能是为编译程序提供非常规的控制流信息。宏替换: #define,这是最常见的用法,它可以定义符号常量、函数功能、重新命名、字符串的拼接等各种功能。三、预处理指令:预处理指令的格式如下:# directive 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;# elseenum { is_named = is_named_parameter<Value>::value };typedef typename if_true<(is_named)>::templatethen<make_named_arg, make_key_value>::type Make;# endif下面我们看一下常见的预处理指令:#define 宏定义#undef 未定义宏#include 文本包含#ifdef 如果宏被定义就进行编译#ifndef 如果宏未被定义就进行编译#endif 结束编译块的控制#if 表达式非零就对代码进行编译#else 作为其他预处理的剩余选项进行编译#elif 这是一种#else和#if的组合选项 //后面有例子的498)this.style.width=498;" border=0>#line 改变当前的行数和文件名称#error 输出一个错误信息#pragma 为编译程序提供非常规的控制流信息下面我们对这些预处理进行一一的说明,考虑到宏的重要性和繁琐性,我们把它放到最后讲。四、文件包含指令:这种预处理使用方式是最为常见的,平时我们编写程序都会用到,最常见的用法是:#include <iostream> //标准库头文件#include <iostream.h> //旧式的标准库头文件#include "IO.h" //用户自定义的头文件#include "../file.h" //UNIX下的父目录下的头文件#include "/usr/local/file.h" //UNIX下的完整路径#include "../file.h" //Dos下的父目录下的头文件#include "/usr/local/file.h" //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 identifieryour code#endif如果identifier为一个定义了的符号,your code就会被编译,否则剔除2、#ifndef identifieryour code#endif如果identifier为一个未定义的符号,your code就会被编译,否则剔除3、#if expressionyour code#endif如果expression非零,your code就会被编译,否则剔除4、#ifdef identifieryour code1#elseyour code2#endif如果identifier为一个定义了的符号,your code1就会被编译,否则yourcode2就会被编译5、#if expressin1your code1#elif expression2 //呵呵,elifyour code2#elseyour 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是非统一的,他要依靠各个编译器生产者,例如,在SUN C++编译器中:// 把name和val的起始地址调整为8个字节的倍数#progma align 8 (name, val)char name[9];double val;//在程序执行开始,调用函数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;预处理何去何从如何取代#include预处理指令,我们在这里就不再一一讨论了。C++并没有为#include提供替代形式,但是namespace提供了一种作用域机制,它能以某种方式支持组合,利用它可以改善#include的行为方式,但是我们还是无法取代#include。#progma应该算是一个可有可无的预处理指令,按照C++之父Bjarne的话说,就是:"#progma被过分的经常的用于将语言语义的变形隐藏到编译系统里,或者被用于提供带有特殊语义和笨拙语法的语言扩充。”对于#ifdef,我们仍然束手无策,就算是我们利用if语句和常量表达式,仍然不足以替代她,因为一个if语句的正文必须在语法上正确,满足类检查,即使他处在一个绝不会被执行的分支里面。

分享到:
评论

相关推荐

    C++编译预处理 C++编译预处理

    C++编译预处理,关于C++学习的课件,可以帮助C++的学习

    C++图像预处理及数字识别源代码.rar

    C++图像预处理及数字识别源代码,适合学生

    c++预处理指令

    c++预处理指令是C++语言中的一种机制,允许开发者在编译器中插入特殊的指令,以控制编译器的行为。这些指令可以在不同的机器和操作系统上编译,以保持C和C++完全兼容。 #pragma是C++语言中最复杂的预处理指令,提供...

    C++进行图像预处理代码

    在IT领域,图像预处理是计算机视觉和机器学习项目中的重要步骤。本资源是一个C++实现的图像...通过深入研究和实践这个C++代码库,开发者不仅可以提升技术技能,还能更好地理解和利用图像预处理在实际问题中的作用。

    c/c++词法分析预处理

    c/c++词法分析预处理cpp源代码,将同目录下"abc.cpp"去除跳格回车换行注释和不必要的空格,输出文件"abc_chuli.cpp",因为具体要求不同,这处理出来的文件还是能继续编译运行。

    C++写Arff数据预处理

    至于压缩包中的文件名“WekaPreprocess”,这可能是指一个C++库或工具,用于与Weka接口,实现数据预处理和K-Means聚类。Weka是一个强大的Java工具箱,提供了多种数据预处理方法和机器学习算法,包括K-Means。通过C++...

    C_C++ 语言参考 基本C_C++ 预处理命令

    本文将深入探讨"基本C/C++预处理命令",包括操作符优先级、转义字符、ASCII码表、基本数据类型以及关键字,并简要提及标准C库。 首先,预处理命令是C/C++程序编译过程中的第一步,它们不直接影响最终的机器代码,但...

    C/C++预处理分析与改进 .PDF

    C/C++预处理分析与改进 .PDF

    C++ 函数与预处理

    【函数与预处理】是C++编程中的重要概念,它们是构建复杂程序结构的基础。在C++中,函数是一组完成特定任务的代码块,可以被多次调用,从而实现代码复用。预处理则是编译过程的早期阶段,涉及宏定义、条件编译等操作...

    C++函数与预处理教程

    在C++编程语言中,函数和预处理是两个至关重要的概念。函数是代码组织的基本单元,它们封装了特定的功能,可以被多次调用,降低了代码的复杂性。预处理则是编译过程的一部分,用于处理源代码中的特定指令,如宏定义...

    语音信号预处理全过程c++编写

    本文将详细介绍使用C++进行语音信号预处理的全过程,包括分帧、预加重、断点检测等关键步骤,并以实际代码示例进行解析。 1. 分帧 在处理连续的语音信号时,由于信号长度不定,我们需要将其分割成固定长度的小段,...

    ISO-ANSI C标准译文与注解 C-C++预处理部分.chm

    这个CHM文件有问题,您不要再下载它了. 请下载下面这个HTML格式的: http://download.csdn.net/source/476473

    c程序,脑电数据处理,包括预处理,能量特征提取,fisher分类

    标题中的"C程序,脑电数据处理,包括预处理,能量特征提取,Fisher分类"涉及的是使用C语言进行脑电信号(EEG)处理的项目。这个项目包含了数据的获取、预处理、特征提取以及使用Fisher算法进行分类等多个关键步骤,...

    msc.rar_MSC光谱预处理_msc算法_光谱_光谱 预处理_光谱处理 C++

    本文将详细探讨“msc.rar”压缩包中涉及到的“MSC光谱预处理”及其核心算法——“MSC算法”,以及如何利用C++进行光谱预处理。 MSC,全称为Multiplicative Scatter Correction(乘性散射校正),是一种常见的光谱...

    ISO-ANSI C标准译文与注解 C-C++预处理部分

    ### ISO-ANSI C标准译文与注解 C-C++预处理部分 #### ISO-ANSI C标准概览 ISO-ANSI C标准是国际标准化组织(ISO)和美国国家标准协会(ANSI)共同制定的一套C语言规范,旨在为C语言提供一个统一的标准框架,确保...

Global site tag (gtag.js) - Google Analytics