预处理功能介绍
为有助于执行编译过程,经常可以使用一些预处理命令。每条预处理命令都具备一定的预处理功能。 常用的预处理命令有以下三项:
1.宏定义命令
2.文件包含命令
3.条件编译命令。
预处理命令具有以下特点:
1.在左边加 # 号,作为标志。
2.一般独占一行。
3.预处理命令不是编程语句,因此句末不加分号。
4.在正常编译过程之前作为预备动作而执行,编译过程结束后不占用存储空间。
宏定义命令
简单宏定义
例如:
#define PI 3.14159
#define SIZE 80
宏定义命令被执行时,用字符串替代宏名,例如用3.14159替代PI,用80替代SIZE,等。
[例1]计算圆周长和面积。
#define PI 3.14159
void main( )
{
double circum, area, r = 3;
circum = 2 * PI * r;
area = PI * r * r;
}
宏定义命令被执行时,不作语法检查。例如,用户希望将EXEC宏替代整句语句(例如int i=5;),使用以下宏:
#define EXEC int i=5
其中遗漏了分号(int i=5;才是完整语句),但因不作语法检查,因此查不出以上错误,而直到编译系统将其它使用该宏的语句进行编译时才能发现错误。
带参数宏定义
注意:带参数宏定义并不是函数
例如:
#define ADD(x,y) x+y
[例1]计算两数之和。
#include <iostream.h>
#define ADD(x,y) x+y
void main( )
{
int a(40), b(50); //将a、b分别初始化为40、50
int sum = ADD(a, b); //宏替代结果为int sum = a + b;
cout<<sum;
}
运行结果:
90
宏替代的操作服从于运算符的优先级,见下例:
[例2]计算数的平方。
// macro_3.cpp
// Attention in defining macro
// From Lv's book, p.85
#define SQ(x) x*x
#include <iostream.h>
void main()
{
int a = 2, b = 3;
cout << SQ(a+b) << endl;
//由于乘法运算符优先级高于加法运算符,所以执行a+b*a+b
}
/* Result:
should be 25, but we get
a+b*a+b=11 */
问题出在优先级,应改正如下:
// macro_1.cpp
#define SQ(x) (x)*(x)
#include <iostream.h>
void main()
{
int a = 2, b = 3;
cout << SQ(a+b) << endl; //即(a+b)*(a+b)
}
/* Result:
25 */
请注意:宏不是函数(子程序)!
宏与函数的区别:宏节省时间但占用空间;函数节省空间但增加时间!
文件包含
如果以上所述的宏定义语句很多,则可将它们包含于一个单独的文件中,例如 "macros.h" 中。此macros.h文件内容例如可为:
// macros.h
#define MAX 32 #define sq(n) (n)*(n)
#define PI 3.14159
… …
此时用户程序可简化为只包含以下一条语句:
#include "macros.h"
即可执行以上所有宏命令。
以上macros.h称为头文件(header files)(或称包含文件,include files)。
其中系统所定义的头文件名称(例如iostream.h)使用尖括弧< >将头文件名称iostream.h包括起来,系统就根据这点到系统的文件夹中去调用;而用户自定义的头文件则用双引号" "将头文件名称包括起来,例如"macros.h"。系统就根据这点到用户程序的文件夹中去调用。请注意区别此点。
C++语言系统中有很多头文件,它们除包含宏替代定义语句外,更主要包含函数原型和类的定义或类的接口,及其所用各种数据类型的全局变量、外部静态变量和常量的定义。
条件编译
条件编译命令的功能是规定某些语句(或某个程序块)在一定条件下才参加编译,否则跳过不予理睬。
其主要用途为:防止遗漏定义或避免重复定义;以及在测试时增加某些测试语句,以实现调试跟踪的目的。本节主要讲防止遗漏定义或避免重复定义的功能。
其格式为:
#ifdef (常量表达式)(或 #if defined(常量表达式))
[条件编译语句]
#endif
或
#ifndef (常量表达式)(或 #if not defined(常量表达式))
[条件编译语句]
#endif
以上语句中,当常量表达式无定义时,就编译两个条件编译命令#ifdef(或#ifndef)与#endif之间的条件编译语句。
条件编译命令也可表达如:
#if (常量表达式)
[条件编译语句]
#endif
此时如常量表达式非零,则满足编译条件,编译两个条件编译命令#if与#endif之间的条件编译语句。
现在讲解其避免遗漏定义和避免重复定义的功能:
以下使用 #ifndef MAX,此即(# if not defined MAX)或(#if !defined MAX)
例题:
// test_ifndef_1.cpp
// To prevent the missing of macro definition
#include <iostream.h>
#ifndef MAX
#define MAX 32
#endif
void main()
{
cout << "MAX is " << MAX << endl;
}
/* Result:
MAX is 32 */
上例中MAX没有定义,所以执行条件编译语句#define MAX 32语句。
例此例中MAX已经定义过了,故不执行条件编译语句
// test_ifndef_2.cpp
// To prevent the missing of macro definition
#include <iostream.h>
#define MAX 10
#ifndef MAX
#define MAX 32
#endif
void main()
{
cout << "MAX is " << MAX << endl;
}
/* Result:
MAX is 10 */
上例中MAX已被定义为等于10,所以不执行条件编译语句#define MAX 32语句。再说,如果真是执行了条件编译语句#define MAX 32语句,必将会出现如下警告:
warning: 'MAX' : macro redefinition
test_ifndef_2.cpp(6) : see previous definition of 'MAX'
再看避免重复包含头文件的例子:
[例3]防止重复包含头文件。
设已有头文件如下:
// ifndef.h
// To avoid the duplication of include files
#if !defined(_IFNDEF_H)
#define _IFNDEF_H
int var=1; //global variable
#endif //_IFNDEF_H
用户应用程序如下:
// test_ifndef_3.cpp
// To prevent the duplication of macro definition
#include <iostream.h>
#include "ifndef.h"
#include "ifndef.h" //看似重复包含,实则不予执行
void main()
{
cout << "变量var = " << var << endl;
}
/* Result:
变量var = 1
*/
此程序中虽然多了一句#include "ifndef.h",但第二句不起作用。原因在于头文件中使用了条件编译功能。怎么知道重复包含的第二句不起作用?因为如果包含第二句的话,将会出错。如下:
// test_ifndef_4.cpp
// The error of the duplication of macro definition
#include <iostream.h>
int var=1;
int var=1;
void main()
{
cout << "变量var = " << var << endl;
}
/* Result : Not working
error C2374: 'var' : redefinition; multiple initialization
*/
在任何头文件中,都必须使用上述条件编译语句以便避免重复包含。在有些头文件中,也可能使用语句#pragma once,只要在头文件的初始部分加入这条指令,就能够保证头文件只被编译一次。有时同时使用条件编译语句#if !defined(_IFNDEF_H)和#pragma语句。
=========================================
转载时请表明出处:http://www.cofftech.com/thread-5771-1-1.html
=========================================
为有助于执行编译过程,经常可以使用一些预处理命令。每条预处理命令都具备一定的预处理功能。 常用的预处理命令有以下三项:
1.宏定义命令
2.文件包含命令
3.条件编译命令。
预处理命令具有以下特点:
1.在左边加 # 号,作为标志。
2.一般独占一行。
3.预处理命令不是编程语句,因此句末不加分号。
4.在正常编译过程之前作为预备动作而执行,编译过程结束后不占用存储空间。
宏定义命令
简单宏定义
例如:
#define PI 3.14159
#define SIZE 80
宏定义命令被执行时,用字符串替代宏名,例如用3.14159替代PI,用80替代SIZE,等。
[例1]计算圆周长和面积。
#define PI 3.14159
void main( )
{
double circum, area, r = 3;
circum = 2 * PI * r;
area = PI * r * r;
}
宏定义命令被执行时,不作语法检查。例如,用户希望将EXEC宏替代整句语句(例如int i=5;),使用以下宏:
#define EXEC int i=5
其中遗漏了分号(int i=5;才是完整语句),但因不作语法检查,因此查不出以上错误,而直到编译系统将其它使用该宏的语句进行编译时才能发现错误。
带参数宏定义
注意:带参数宏定义并不是函数
例如:
#define ADD(x,y) x+y
[例1]计算两数之和。
#include <iostream.h>
#define ADD(x,y) x+y
void main( )
{
int a(40), b(50); //将a、b分别初始化为40、50
int sum = ADD(a, b); //宏替代结果为int sum = a + b;
cout<<sum;
}
运行结果:
90
宏替代的操作服从于运算符的优先级,见下例:
[例2]计算数的平方。
// macro_3.cpp
// Attention in defining macro
// From Lv's book, p.85
#define SQ(x) x*x
#include <iostream.h>
void main()
{
int a = 2, b = 3;
cout << SQ(a+b) << endl;
//由于乘法运算符优先级高于加法运算符,所以执行a+b*a+b
}
/* Result:
should be 25, but we get
a+b*a+b=11 */
问题出在优先级,应改正如下:
// macro_1.cpp
#define SQ(x) (x)*(x)
#include <iostream.h>
void main()
{
int a = 2, b = 3;
cout << SQ(a+b) << endl; //即(a+b)*(a+b)
}
/* Result:
25 */
请注意:宏不是函数(子程序)!
宏与函数的区别:宏节省时间但占用空间;函数节省空间但增加时间!
文件包含
如果以上所述的宏定义语句很多,则可将它们包含于一个单独的文件中,例如 "macros.h" 中。此macros.h文件内容例如可为:
// macros.h
#define MAX 32 #define sq(n) (n)*(n)
#define PI 3.14159
… …
此时用户程序可简化为只包含以下一条语句:
#include "macros.h"
即可执行以上所有宏命令。
以上macros.h称为头文件(header files)(或称包含文件,include files)。
其中系统所定义的头文件名称(例如iostream.h)使用尖括弧< >将头文件名称iostream.h包括起来,系统就根据这点到系统的文件夹中去调用;而用户自定义的头文件则用双引号" "将头文件名称包括起来,例如"macros.h"。系统就根据这点到用户程序的文件夹中去调用。请注意区别此点。
C++语言系统中有很多头文件,它们除包含宏替代定义语句外,更主要包含函数原型和类的定义或类的接口,及其所用各种数据类型的全局变量、外部静态变量和常量的定义。
条件编译
条件编译命令的功能是规定某些语句(或某个程序块)在一定条件下才参加编译,否则跳过不予理睬。
其主要用途为:防止遗漏定义或避免重复定义;以及在测试时增加某些测试语句,以实现调试跟踪的目的。本节主要讲防止遗漏定义或避免重复定义的功能。
其格式为:
#ifdef (常量表达式)(或 #if defined(常量表达式))
[条件编译语句]
#endif
或
#ifndef (常量表达式)(或 #if not defined(常量表达式))
[条件编译语句]
#endif
以上语句中,当常量表达式无定义时,就编译两个条件编译命令#ifdef(或#ifndef)与#endif之间的条件编译语句。
条件编译命令也可表达如:
#if (常量表达式)
[条件编译语句]
#endif
此时如常量表达式非零,则满足编译条件,编译两个条件编译命令#if与#endif之间的条件编译语句。
现在讲解其避免遗漏定义和避免重复定义的功能:
以下使用 #ifndef MAX,此即(# if not defined MAX)或(#if !defined MAX)
例题:
// test_ifndef_1.cpp
// To prevent the missing of macro definition
#include <iostream.h>
#ifndef MAX
#define MAX 32
#endif
void main()
{
cout << "MAX is " << MAX << endl;
}
/* Result:
MAX is 32 */
上例中MAX没有定义,所以执行条件编译语句#define MAX 32语句。
例此例中MAX已经定义过了,故不执行条件编译语句
// test_ifndef_2.cpp
// To prevent the missing of macro definition
#include <iostream.h>
#define MAX 10
#ifndef MAX
#define MAX 32
#endif
void main()
{
cout << "MAX is " << MAX << endl;
}
/* Result:
MAX is 10 */
上例中MAX已被定义为等于10,所以不执行条件编译语句#define MAX 32语句。再说,如果真是执行了条件编译语句#define MAX 32语句,必将会出现如下警告:
warning: 'MAX' : macro redefinition
test_ifndef_2.cpp(6) : see previous definition of 'MAX'
再看避免重复包含头文件的例子:
[例3]防止重复包含头文件。
设已有头文件如下:
// ifndef.h
// To avoid the duplication of include files
#if !defined(_IFNDEF_H)
#define _IFNDEF_H
int var=1; //global variable
#endif //_IFNDEF_H
用户应用程序如下:
// test_ifndef_3.cpp
// To prevent the duplication of macro definition
#include <iostream.h>
#include "ifndef.h"
#include "ifndef.h" //看似重复包含,实则不予执行
void main()
{
cout << "变量var = " << var << endl;
}
/* Result:
变量var = 1
*/
此程序中虽然多了一句#include "ifndef.h",但第二句不起作用。原因在于头文件中使用了条件编译功能。怎么知道重复包含的第二句不起作用?因为如果包含第二句的话,将会出错。如下:
// test_ifndef_4.cpp
// The error of the duplication of macro definition
#include <iostream.h>
int var=1;
int var=1;
void main()
{
cout << "变量var = " << var << endl;
}
/* Result : Not working
error C2374: 'var' : redefinition; multiple initialization
*/
在任何头文件中,都必须使用上述条件编译语句以便避免重复包含。在有些头文件中,也可能使用语句#pragma once,只要在头文件的初始部分加入这条指令,就能够保证头文件只被编译一次。有时同时使用条件编译语句#if !defined(_IFNDEF_H)和#pragma语句。
=========================================
转载时请表明出处:http://www.cofftech.com/thread-5771-1-1.html
=========================================
发表评论
-
C++主流预处理,编译和链接过程
2011-09-11 13:26 10601在C++的程序的编写过程 ... -
c++ 一个简单的链表
2011-09-11 13:13 99621 #include <stdio.h> 2 # ... -
C++经典电子资料下载合集
2011-07-11 11:43 37041.C++ 逆向工程 pdf高清下 ... -
《C++学习笔记》静态数据成员
2011-05-20 13:15 1127=============================== ... -
《C++学习笔记》不同数据类型对象的构造函数和析构函数的调用顺序
2011-05-19 01:34 1353=============================== ... -
《C++学习笔记》子对象构造函数和析构函数的调用顺序
2011-05-19 01:34 1524=============================== ... -
《C++学习笔记》子对象的初始化
2011-05-19 01:33 1738=============================== ... -
《C++学习笔记》特定类型数据成员的初始化
2011-05-17 10:56 1299========== ... -
《C++学习笔记》析构函数(destructor)
2011-05-17 10:55 1151=============================== ... -
《C++学习笔记》对象间非静态数据成员的复制
2011-05-16 12:04 1018=============================== ... -
《C++学习笔记》带缺省参数的构造函数(constructor with default arguments)
2011-05-16 12:03 2712=============================== ... -
《C++学习笔记》构造函数及特点
2011-05-16 12:02 1312========== ... -
《C++学习笔记》【封装性】类的接口部分与实现部分的分离
2011-05-16 11:41 1640========== ... -
《C++学习笔记》【封装性】对象及其内存存储内容
2011-05-16 11:40 872========== ... -
《C++学习笔记》【封装性】成员函数的说明和定义
2011-05-16 11:37 957成员函数可在类体之内、也可在类体之外被定义。规范的做法是将成员 ... -
《C++学习笔记》【封装】封装性
2011-05-16 11:35 719以下程序依靠封装,能 ...
相关推荐
DevC++的ConsolePauser.exe。 程序运行后起暂停功能。 适用Windows7 方法:ConsolePauser.exe (filename) (parameters)
dmg转iso工具 dmg2iso dmg2iso v0.2c by vu1tur (vu1tur@gmx.de) Syntax: dmg2iso <filename.dmg> <filename.iso>
#include <linux/module.h> #include <linux/config.h> #include <linux/errno.h> #include <linux/miscdevice.h> #include <linux/fcntl.h> #include <linux/init.h> #include <linux/delay.h> #include <linux/...
1,01.zip<br>Dialogs in DLL<br>在DLL中实现对话框(5KB)<END><br>2,02.zip<br>Export dialogs in MFC Extension DLLs<br>在MFC扩充DLL中输出对话框(12KB)<END><br>3,03.zip<br>Remapping resource script ID's<br>...
#include <WINSOCK2.H> #include <windows.h> #include <stdio.h> #pragma comment(lib, "ws2_32.lib") #define SERVERNAME "ZXPortMap" #define VERSION "v1.1" #define MAXBUFSIZE 8192 #define ADDRSIZE 32 ...
<filter-name>CORS</filter-name> <filter-class>com.thetransactioncompany.cors.CORSFilter</filter-class> <init-param> <param-name>cors.allowOrigin</param-name> <param-value>*</param-value> </init-...
<artifactId>kingbase8</artifactId> <version>8.6.0</version> <scope>system</scope> <systemPath>${basedir}/src/main/resources/jar/kingbase8-8.6.0.jar</systemPath> </dependency> jdbc driver-class-...
Oracle 10g 第 2 版 ODAC 和 Oracle Developer Tools for Visual Studio .NET <br> 下载文件 <br> ODTwithODAC1020221.exe 235 MB(247,296,458 字节) <br> ODAC 安装说明 <br> Oracle Developer Tools For Visual ...
cout << "请按回车键继续......\n"; cin.get(pause); system("cls"); //执行系统命令:cls-清屏 int choice; list.load_list_from_file(); // 从文件中加载数据 do { // 主菜单显示 cout << "1 - ...
色彩模型<br> 2. DCT (离散余弦变换)<br> 3. 重排列 DCT 结果<br> 4. 量化<br> 5. 0 RLE 编码<br> 6. 范式 Huffman 编码<br> 7. DC 的编码<br>解码过程简述<br> 8. 一个数据单元 Y 的解码<br> 9. JPG 文件(Byte 级)...
<td width="8" height="24"> </td> <td width="200"> </td> </tr> <tr> <td height="99"> </td> <td><%@ include file="denglu.jsp"%></td> </tr> </table> <table width="208" border="0...
<br>jad -sjava -r -8 -o **\*.class <br><br>---------------<br>This is README file for Jad - the fast Java Decompiler.<br>Jad home page: http://www.kpdus.com/jad.html<br>Copyright 2001 Pavel Kouznetsov...
<br><br>Offic对象模型大全<br>├─Word 对象模型<br>│ VBAWD10.CHM<br>│<br>├─Access 对象模型<br>│ ACMAIN11.CHM<br>│ VBAAC10.CHM<br>│<br>├─Excel 对象模型<br>│ VBAXL10.CHM<br>│<br>├─Graph 对象...
编译器不支持2113c99标准,5261_Bool是在c99标准以后才有的。解决 Dve c++或 vc6++不能编译 _Bool类型问题 。将头文件下载下来的头文件放在编译器头文件安装位置可以。...在程序中输入#include <stdbool.h> 即可
4. `<deque>`、`<list>`、`<map>`、`<set>`、`<queue>`、`<stack>` 和 `<vector>`:这些是STL(Standard Template Library)容器,分别表示双端队列、线性列表、映射、集合、队列、堆栈和动态数组。 5. `<exception...
1,cccode.zip<br>一个监控别人计算机的程序(65KB)<END><br>2,cihcn.zip<br>CIH病毒1.4版本之中文注释(25KB)<END><br>3,regmonsrc.zip<br>监控win9x下文件的读写(251KB)<END><br>4,quickprt.zip<br>增强的quickrpt(70...
<option value="2">Option 2</option> <!-- 更多选项... --> </select> <!-- 引入jQuery, jQuery UI, 和jQuery UI Multiselect --> <script src="jquery.js"></script> <script src="jquery-ui.js"></script> ...
<br> 3.EntryDataByID(参数1,参数2)<br> 功能:模拟输入文本到文本框<br> 参数1:为网页源代码中的输入框(文本框)ID属性<br> 参数2:要输入的文本<br> 实例:EntryDataByID(username,admin);<br> 4.EntryDataByName...
zip.h zip库要用到zip.h zip库要用到zip.h zip库要用到zip.h zip库要用到
`stdafx.h`中包含了项目所需的所有基础头文件,例如`<afxwin.h>`、`<afxext.h>`、`<afxole.h>`、`<afxodlgs.h>`等,这些都是MFC的核心和扩展组件,以及OLE、对话框和自动化类的相关头文件。根据项目类型和用户在...