南昌网站设计公司哪家好,php网站开发机试题目,专业做根雕的网站,如何做外文网站C标准中的一些预定义宏昨天写代码时需要在代码获取当前编译时间#xff0c;从而可动态地作为版本信息#xff0c;因此用到了C标准中的一些预定义的宏。在此将C标准中定义的几个宏一并总结一下#xff1a; __DATE__ 进行预处理的日期#xff08;“Mmm dd yyyy”形式的字符串…C标准中的一些预定义宏昨天写代码时需要在代码获取当前编译时间从而可动态地作为版本信息因此用到了C标准中的一些预定义的宏。在此将C标准中定义的几个宏一并总结一下 __DATE__ 进行预处理的日期“Mmm dd yyyy”形式的字符串文字如May 27 2006 __FILE__ 代表当前源代码文件名的字符串文字 包含了详细路径如G:/program/study/c/test1.c __LINE__ 代表当前源代码中的行号的整数常量 __TIME__ 源文件编译时间格式微“hhmmss”如09:11:10 __func__ 当前所在函数名在编译器的较高版本中支持 __FUNCTION__ 当前所在函数名 对于__FILE____LINE____func____FUNCTION__ 这样的宏在调试程序时是很有用的因为你可以很容易的知道程序运行到了哪个文件的那一行是哪个函数。 而对于__DATE____TIME__则可以获取编译时间如如下代码通过宏获取编译时间并通过sscanf从中获取具体的年月日时分秒数据可在代码中做相应使用。我的代码中是根据此数据作为版本标识并依此判断哪个版本新些及是否需要升级。 char * creationDate __DATE__ , __TIME__; sscanf(creationDate, %s %d %d, %d:%d:%d, month, day, year, hour, min, sec); 预处理命令#pragma和预定义宏--转载 一、C预定义宏 C标准指定了一些预定义宏,编程中常常用到。 __DATE__ 进行预处理的日期 __FILE__ 代表当前源代码文件名的字符串 __LINE__ 代表当前源代码文件中行号的整数常量 __STDC__ 设置为1时,表示该实现遵循C标准 __STDC_HOSTED__ 为本机环境设置为,否则设为0 __STDC_VERSION__ 为C99时设置为199901L __TIME__ 源文件的编译时间 __func__ C99提供的为所在函数名的字符串 对于__FILE____LINE____func__这样的宏在调试程序时是很有用的因为你可以很容易的知道程序运行到了哪个文件的那一行是哪个函数. 例如: #include #include void why_me(); int main() { printf( The file is %s/n, __FILE__ ); printf( The date is %s/n, __DATE__ ); printf( The time is %s/n, __TIME__ ); printf(The version is %s/n,__STDC__VERSION__); printf( This is line %d/n, __LINE__ ); printf( This function is %s/n , __func__ ); why_me(); return 0; } void why_me() { printf( This function is %s/n, __func__ ); printf( This is line %d/n, __LINE__ ); } 二、#line和#error #line用于重置由__LINE__和__FILE__宏指定的行号和文件名。 用法如下#line number filename 例如#line 1000 //将当前行号设置为1000 #line 1000 lukas.c //行号设置为1000,文件名设置为lukas.c #error指令使预处理器发出一条错误消息该消息包含指令中的文本.这条指令的目的就是在程序崩溃之前能够给出一定的信息。 三、#pragma 在所有的预处理指令中#Pragma 指令可能是最复杂的了。#pragma的作用是设定编译器的状态或者是指示编译器完成一些特定的动作。#pragma指令对每个编译器给出了一个方法,在保持与C和C语言完全兼容的情况下,给出主机或操作系统专有的特征。依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的。 其格式一般为: #Pragma Para 其中Para 为参数下面来看一些常用的参数。 (1)message 参数。 Message 参数是我最喜欢的一个参数它能够在编译信息输出窗口中输出相应的信息这对于源代码信息的控制是非常重要的。其使用方法为 #Pragma message(“消息文本”) 当编译器遇到这条指令时就在编译输出窗口中将消息文本打印出来。 当我们在程序中定义了许多宏来控制源代码版本的时候我们自己有可能都会忘记有没有正确的设置这些宏此时我们可以用这条指令在编译的时候就进行检查。假设我们希望判断自己有没有在源代码的什么地方定义了_X86这个宏可以用下面的方法 #ifdef _X86 #Pragma message(“_X86 macro activated!”) #endif 当我们定义了_X86这个宏以后应用程序在编译时就会在编译输出窗口里显示“_ X86 macro activated!”。我们就不会因为不记得自己定义的一些特定的宏而抓耳挠腮了。 (2)另一个使用得比较多的pragma参数是code_seg。格式如 #pragma code_seg( [section-name[,section-class] ] ) 它能够设置程序中函数代码存放的代码段当我们开发驱动程序的时候就会使用到它。 (3)#pragma once (比较常用 只要在头文件的最开始加入这条指令就能够保证头文件被编译一次。这条指令实际上在VC6中就已经有了但是考虑到兼容性并没有太多的使用它。 (4)#pragma hdrstop表示预编译头文件到此为止后面的头文件不进行预编译。BCB可以预编译头文件以加快链接的速度但如果所有头文件都进行预编译又可能占太多磁盘空间所以使用这个选项排除一些头文件。 有时单元之间有依赖关系比如单元A依赖单元B所以单元B要先于单元A编译。你可以用#pragma startup指定编译优先级如果使用了#pragma package(smart_init) BCB就会根据优先级的大小先后编译。 (5)#pragma resource *.dfm表示把*.dfm文件中的资源加入工程。*.dfm中包括窗体外观的定义。 (6)#pragma warning( disable : 4507 34; once : 4385; error : 164 )等价于 #pragma warning(disable:4507 34) /* 不显示4507和34号警告信息。如果编译时总是出现4507号警告和34号警告 而认为肯定不会有错误可以使用这条指令。*/ #pragma warning(once:4385) // 4385号警告信息仅报告一次 #pragma warning(error:164) // 把164号警告信息作为一个错误。 同时这个pragma warning 也支持如下格式 #pragma warning( push [ ,n ] ) #pragma warning( pop ) 这里n代表一个警告等级(1---4)。 #pragma warning( push )保存所有警告信息的现有的警告状态。 #pragma warning( push, n)保存所有警告信息的现有的警告状态并且把全局警告等级设定为n。 #pragma warning( pop )向栈中弹出最后一个警告信息在入栈和出栈之间所作的一切改动取消。例如 #pragma warning( push ) #pragma warning( disable : 4705 ) #pragma warning( disable : 4706 ) #pragma warning( disable : 4707 ) //....... #pragma warning( pop ) 在这段代码的最后重新保存所有的警告信息(包括47054706和4707)。 7pragma comment(...) 该指令将一个注释记录放入一个对象文件或可执行文件中。 常用的lib关键字可以帮我们连入一个库文件。 8progma packn 指定结构体对齐方式#pragma pack(n)来设定变量以n字节对齐方式。n 字节对齐就是说变量存放的起始地址的偏移量有两种情况第一、如果n大于等于该变量所占用的字节数那么偏移量必须满足默认的对齐方式第二、如果n小于该变量的类型所占用的字节数那么偏移量为n的倍数不用满足默认的对齐方式。结构的总大小也有个约束条件分下面两种情况如果n大于所有成员变量类型所占用的字节数那么结构的总大小必须为占用空间最大的变量占用的空间数的倍数 否则必须为n的倍数。下面举例说明其用法。 #pragma pack(push) //保存对齐状态 #pragma pack(4)//设定为4字节对齐 struct test { char m1; double m4; int m3; }; #pragma pack(pop)//恢复对齐状态 为测试该功能可以使用sizeof测试结构体的长度 在你写dll的时候,因为对于C和C,编译器会有不同的名字解析规则,所以可以这样用 #ifndef __STDC__ extern C void function(); #else void function(); #endif __LINE__ 在源代码中插入当前源代码行号 __FILE__ 在源代码中插入当前源代码文件名 __DATE__ 在源代码中插入当前编译日期〔注意和当前系统日期区别开来〕 __TIME__ 在源代码中插入当前编译时间〔注意和当前系统时间区别开来〕 __STDC__ 当要求程序严格遵循ANSIC标准时该标识符被赋值为1。 ---------------------------------------------------------------------------- 标识符__LINE__和__FILE__通常用来调试程序标识符__DATE__和__TIME__通常用来在编译后的程序中加入一个时间标志以区分程序的不同版本当要求程序严格遵循ANSIC标准时标识符__STDC__就会被赋值为1当用C编译程序编译时标识符__cplusplus就会被定义。 #include int main () { printf(该输出行在源程序中的位置%d/n, __LINE__ ); printf(该程序的文件名为%s/n, __FILE__ ); printf(当前日期为%s/n, __DATE__ ); printf(当前时间为%s/n, __TIME__ ); return 0; } #include void main(void) { printf(%d,__LINE__); // Line 5 } 结果为:5 // 标准预定义宏宏.cpp : Defines the entry point for the console application. // #include stdafx.h #include void main(void) { printf(%d,__LINE__); // Line 5 } 编译器宏使用总结 C/C中宏总结C程序的源代码中可包括各种编译指令这些指令称为预处理命令。虽然它们实际上不是C语言的一部分但却扩展了C程序设计的环境。本节将介绍如何应用预处理程序和注释简化程序开发过程并提高程序的可读性。ANSI标准定义的c语言预处理程序包括下列命令 #define#errorinclude#if#else#elif#endif#ifdef#ifndef#undef#line#pragma等。非常明显所有预处理命令均以符号#开头下面分别加以介绍。 命令#define定义了一个标识符及一个串。在源程序中每次遇到该标识符时均以定义的串代换它。ANSI标准将标识符定义为宏名将替换过程称为宏替换。命令的一般形式为 #define identifier string 注意 ? 该语句没有分号。在标识符和串之间可以有任意个空格串一旦开始仅由一新行结束。 ? 宏名定义后即可成为其它宏名定义中的一部分。 ? 宏替换仅仅是以文本串代替宏标识符前提是宏标识符必须独立的识别出来否则不进行替换。例如 #define XYZ this is a tes 使用宏printf(XYZ)//该段不打印this is a test而打印XYZ。因为预编译器识别出的是XYZ ? 如果串长于一行可以在该行末尾用一反斜杠 /续行。 处理器命令#error强迫编译程序停止编译主要用于程序调试。 #include 命令i nclude使编译程序将另一源文件嵌入带有i nclude的源文件被读入的源文件必须用双引号或尖括号括起来。例如 i ncludestdio.h或者i nclude 这两行代码均使用C编译程序读入并编译用于处理磁盘文件库的子程序。 将文件嵌入i nclude命令中的文件内是可行的这种方式称为嵌套的嵌入文件嵌套层次依赖于具体实现。 如果显式路径名为文件标识符的一部分则仅在哪些子目录中搜索被嵌入文件。否则如果文件名用双引号括起来则首先检索当前工作目录。如果未发现文件则在命令行中说明的所有目录中搜索。如果仍未发现文件则搜索实现时定义的标准目录。 如果没有显式路径名且文件名被尖括号括起来则首先在编译命令行中的目录内检索。 如果文件没找到则检索标准目录不检索当前工作目录。 条件编译命令 有几个命令可对程序源代码的各部分有选择地进行编译该过程称为条件编译。商业软件公司广泛应用条件编译来提供和维护某一程序的许多顾客版本。 #if、#else#elif及#endif #if的一般含义是如果#if后面的常量表达式为true则编译它与#endif之间的代码否则跳过这些代码。命令#endif标识一个#if块的结束。 #if constant-expression statement sequence #endif 跟在#if后面的表达式在编译时求值因此它必须仅含常量及已定义过的标识符不可使用变量。表达式不许含有操作符sizeofsizeof也是编译时求值。 #else命令的功能有点象C语言中的else#else建立另一选择在#if失败的情况下。注意# else属于# if块。 #elif命令意义与ELSE IF 相同它形成一个if else-if阶梯状语句可进行多种编译选择。#elif 后跟一个常量表达式。如果表达式为true则编译其后的代码块不对其它#elif表达式进行测试。否则顺序测试下一块。 #if expression statement sequence #elif expression1 statement sequence #endif 在嵌套的条件编译中#endif、#else或#elif与最近#if或#elif匹配。 # ifdef 和# ifndef 条件编译的另一种方法是用#ifdef与#ifndef命令它们分别表示如果有定义及如果无定义。# ifdef的一般形式是 # ifdef macroname statement sequence #endif #ifdef与#ifndef可以用于#if、#else#elif语句中但必须与一个#endif。 命令#undef 取消其后那个前面已定义过有宏名定义。一般形式为 #undef macroname 命令# line改变__LINE__与__FILE__的内容它们是在编译程序中预先定义的标识符。命令的基本形式如下 # line number[filename] 其中的数字为任何正整数可选的文件名为任意有效文件标识符。行号为源程序中当前行号文件名为源文件的名字。命令# line主要用于调试及其它特殊应用。注意在#line后面的数字标识从下一行开始的数字标识。 预定义的宏名 ANSI标准说明了C中的五个预定义的宏名。它们是 __LINE__ __FILE__ __DATE__ __TIME__ __STDC__ 如果编译不是标准的则可能仅支持以上宏名中的几个或根本不支持。记住编译程序也许还提供其它预定义的宏名。 __LINE__及__FILE__宏指令在有关# line的部分中已讨论这里讨论其余的宏名。 __DATE__宏指令含有形式为月/日/年的串表示源文件被翻译到代码时的日期。 源代码翻译到目标代码的时间作为串包含在__TIME__中。串形式为时分秒。 如果实现是标准的则宏__STDC__含有十进制常量1。如果它含有任何其它数则实现是非标准的。编译C程序时编译器自动定义了一个预处理名字__cplusplus而编译标准C时自动定义名字__STDC__。 注意宏名的书写由标识符与两边各二条下划线构成。 (部分内容出自http://www.bc-cn.NET/Article/kfyy/cyy/jc/200511/919.html) 8、 C、C宏体中出现的#### 宏体中#的功能是将其后面的宏参数进行字符串化操作Stringfication简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。 而##被称为连接符concatenator用来将两个Token连接为一个Token。注意这里连接的对象是Token就行而不一定是宏的变量。比如你要做一个菜单项命令名和函数指针组成的结构体