第9章 预处理命令
- 格式:docx
- 大小:17.05 KB
- 文档页数:3
1.下面叙述中正确的是(A )。
A.带参数的宏定义中参数是没有类型的B.宏展开将占用程序的运行时间C.宏定义命令是C语言中的一种特殊语句D.使用#include命令包含的头文件必须以“.h”为后缀2.在宏定义#define PI3.14159中,用宏名代替一个( D )A.常量B.单精度数C.双精度数D.字符串3.下面有关宏替换的叙述不正确的是( D )A.宏替换不占用运行时间B.宏名无类型C.宏替换只是字符替换D.宏名必须用大写字母表示4.C语言的编译系统对宏命令的处理是( D )A.在程序运行时进行的B.和C程序中的其他语句同时进行编译的C.在程序连接时进行的D.在对源程序中其它成分正式编译之前进行的5.下列定义不正确的是(C )。
A.#define PI 3.141592B.#define S345C.int max(x,y); int x,y; { }D.static char c6.有如下宏定义 #define s(a) a*3 ,则执行表达式x=s(1+2)后,变量x的值为( D)。
A.10B.9C.8D.77.以下说法中正确的是( D)。
A.#define是C语句B.#define是库函数C.#define是自定义函数D.#define是预处理命令8.在“文件包含”预处理语句的使用形式中,当#include后面的文件用了“”(双引号)时,寻找被包含文件的方式是(B)。
A.直接按系统设定的标准方式搜索目录B.先在源程序所在目录搜索,再按系统设定的标准方式搜索C.仅仅搜索源程序所在目录D.仅仅搜索当前目录9.以下叙述正确的是( B )。
A.可以把define和if定义为用户标识符B.可以把define定义为用户标识符,但不能把if定义为用户标识符C.可以把if定义为用户标识符,但不能把define定义为用户标识符D.define和if都不能定义为用户标识符10.以下叙述中正确的是( B )。
#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中就已经有了,但是考虑到兼容性并没有太多的使用它。
.net 预处理详解在使用.net框架进行软件开发时,我们经常会遇到需要在编译阶段对代码进行预处理的情况。
预处理是指在编译代码之前,通过指定一些预处理指令来控制代码的编译过程。
预处理指令在代码中以`#`符号开头,告诉编译器应该如何处理代码。
在本文中,我们将详细介绍.net 框架中常用的预处理指令及其用法。
1. 条件编译条件编译是预处理中最常用的功能之一,它可以根据一些条件来决定是否编译一段特定的代码。
通过使用`#if`和`#endif`指令,我们可以在代码中嵌入任意条件来控制编译过程。
例如,我们可以使用条件编译来根据不同的操作系统平台编译不同的代码片段:```csharp#if WINDOWSConsole.WriteLine("This is a Windows platform.");#elif LINUXConsole.WriteLine("This is a Linux platform.");#endif```上述代码中,`#if`指令用于判断是否为Windows平台,如果是则输出相应的信息;`#elif`指令用于判断是否为Linux平台,如果是则输出相应的信息。
通过条件编译,我们可以根据具体情况选择性地编译代码,以实现更好的跨平台兼容性。
2. 定义常量预处理指令还可以用于定义常量,在编译过程中将其替换为指定的值。
通过`#define`指令可以定义一个常量,并在代码中使用该常量。
例如:```csharp#define MAX_VALUE 100int value = MAX_VALUE;```上述代码中,我们使用`#define`指令定义了一个名为`MAX_VALUE`的常量,并将其设置为100。
之后,在代码中使用该常量时,编译器会将其替换为实际的值。
这样可以提高代码的可读性和维护性。
3. 跳过代码有时候,在调试或者测试代码时,我们需要暂时跳过一些代码片段而不编译它们。
第九章编译预处理课题:第九章编译预处理教学目的:1、了解预处理的概念及特点2、掌握有参宏与无参宏的定义及使用,领会文件包含的使用及效果教学重点:教学难点:掌握宏的使用,文件包含有参宏与无参宏的使用步骤一复习引导ANSI C标准规定可以在C源程序中加入一些“预处理命令”,以改进程序设计环境,提高编程效率。
这些预处理命令是由ANSI C统一规定的,但它不是C语言本身的组成部分,不能直接对它们进行编译。
必须在对程序进行通常的编译之前,先对程序中这些特殊的命令进行“预处理”,即根据预处理命令对程序作相应的处理。
经过预处理后程序不再包括预处理命令了,最后再由编译程序对预处理后的源程序进行通常的编译处理,得到可供执行的目标代码。
步骤二讲授新课C语言与其他高级语言的一个重要区别是可以使用预处理命令和具有预处理的功能。
C 提供的预处理功能主要有以下三种:宏定义、文件包含、条件编译。
分别用宏定义命令、文件包含命令、条件编译命令来实现。
为了与一般C语句相区别,这些命令以符号“ #” 开头。
§9.1宏定义宏:代表一个字符串的标识符。
宏名:被定义为“宏”的标识符。
宏代换(展开):在编译预处理时,对程序中所有出现的“宏名”,用宏定义中的字符串去代换的过程。
一、不带参数的宏定义一般形式:#define 标识符字符串#define PI 3.1415926main(){ float l, s, r, v;printf( “input radius:” );scanf( “%f”, &r );l = 2.0*PI*r;s = PI*r*r;v = 3.0/4*PI*r*r*r;printf(“%10.4f,%10.4f,%10.4\n”, l, s, v);}例如:由键盘输入y值,求表达式:3(y2+3y)+ 4(y2+3y)+ y(y2+3y)#define M (y*y+3*y)main(){ int s, y;printf( “Input a number :”); scanf (“%d”,&y);s=3*M+4*M+y*M; p rintf(“s=%d\n”,s);}先宏展开:s=3*(y*y+3*y) +4*( y*y+3*y) + y*(y*y+3*y)再与源程序合并说明:⑴宏名一般用大写表示,以便与变量名区分。
第9章预处理命令
宏定义不是C语句,所以不能在行尾加分号。
如果加了分号则会连分号一起进行臵换。
可以用#undef命令终止宏定义的作用域。
对程序中用“”括起来的内容(即字符串内的字符),即使与宏名相同,也不进行臵换。
宏定义只做字符替换,不分配内存空间。
宏名不是变量,不分配存储空间,也不能对其进行赋值。
在宏展开时,预处理程序仅对宏名作简单的字符串替换,不作任何检查。
在进行宏定义时,可以引用已定义的宏名
无参宏定义的一般格式:
#define 标识符字符串
将这个标识符(名字)称为“宏名”,在用预编译时将宏名替换成字符串的过程称为“宏展开”。
#define是宏定义命令。
带参宏定义的一般格式:
#define 宏名(形参表)字符串
带参宏的调用和宏展开:
调用格式:宏名(实参表);
宏展开(又称为宏替换)的方法:用宏调用提供的实参直接臵换宏定义中相应的形参,非形参字符保持不变。
定义有参宏时,宏名与左圆括号之间不能留有空格。
否则,C编译系统会将空格以后的所有字符均作为替代字符串,而将该宏视为无参宏。
有参宏的展开,只是将实参作为字符串,简单地臵换形参字符串,而不做任何语法检查。
为了避免出错,可以在所有形参外,甚至整个字符串外,均加上一对圆括号。
如: #define S(r) 3.14*(r)*(r)
则:area=S(a+b); 展开后为: area=3.14*(a+b)*(a+b);
调用有参函数时,是先求出实参的值,然后再复制一份给形参。
而展开有参宏时,只是将实参简单地臵换形参。
函数调用是在程序运行时处理的,为形参分配临时的内存单元;而宏展开则是在编译前进行的,在展开时不分配内存单元,不进行值的传递,也没有“返回值”的概念。
调用函数只可得到一个返回值,而用宏可以设法得到几个结果。
在有参函数中,形参都是有类型的,所以要求实参的类型与其一致;而在有参宏中,形参和宏名都没有类型,只是一个简单的符号代表,因此,宏定义时,字符串可以是任何类型的数据。
使用宏次数多时,宏展开后源程序变长,因为每展开一次都是程序增长,而函数调用不会使源程序变长。
宏替换不占用运行时间,只占编译时间。
而函数调用则占用运行时间(分配单元、保留现场、值传递、返回)。
在程序中如果有带实参的宏,则按#define命令行中指定的字符串从左到右进行臵换。
如果字符串中包含宏中的形参,则将程序语句中相应的实参(可以是常量、变量或表达式)代替形参。
如果宏定义中的字符串中的字符不是参数字符,则保留。
通常如果调用库函数,则用< >,使用用户自己编写的文件,则用“”。
两种格式的区别仅在于:
⑴使用“”:系统首先到当前目录下查找被包含文件,如果没找到,再到系统指定的“包
含文件目录”去查找。
⑵使用< >:直接到系统指定的“包含文件目录”(include)去查找。
预编译时不作任何检查。
只有编译已被宏展开后的源程序时才会发现语法错误并报错。
通常情况下,源程序中所有的行都参加编译,但有时希望对其中一部分内容只在满足一定条件才进行编译,则需要用到条件编译。
(1)一般格式
#ifdef 标识符
程序段1;
#else
程序段2;
#endif
它的作用是若指定的标识符已经被#define命令定义过,则在程序编译阶段编译程序段1,否则,编译程序段2.。
其中,#else部分可以没有。
2)#ifndef 标识符
程序段1
#else
程序段2
#endif
它的作用是若指定的标识符未被#define命令定义过,则在程序编译阶段编译程序段1,否则,编译程序段2.。
(3)#if 表达式
程序段1
#else
程序段2
#endif
它的作用是当指定的表达式的值为真(非零),则在程序编译阶段编译程序段1,否则,编译程序段2.。
(1) 一个#include命令只能指定一个被包含文件,如果要包含n个文件,要用n个#include 命令。
(2) 如果文件1包含文件2,而在文件2中要用到文件3的内容,则可在文件1中用两个include命令分别包含文件2和文件3,而且文件3应出现在文件2之前,即在file1.c中定义。
(3) 在一个被包含文件中又可以包含另一个被包含文件,即文件包含是可以嵌套的。
(4) 在#include命令中,文件名可以用双撇号或尖括号括起来。
(5) 被包含文件(file2.h)与其所在的文件(即用#include命令的源文件file2.c),在预编译后已成为同一个文件(而不是两个文件)。
因此,如果file2.h中有全局静态变量,它也在file1.h 文件中有效,不必用extern声明。
如果需要修改程序中常用的一些参数,可以不必修改每个程序,只需将这些参数放在一个头
文件中,在需要是修改头文件即可,但,被包含文件修改后,凡包含此文件的所有文件都要全部重新编译。