第九章预处理
C语言 第九章 预处理命令
目的: 简化程序的编写 ; 提高程序的模块化、可读性、可移植性。
有三种类型的预处理命令: 1. 宏定义命令; 2. 文件包含命令; 3. 条件编译命令。
为了与C语句区别,这些命令均以“ # ”开头。
处理流程: 第一次编译扫描时,将预编译命令处理完, 然后再进行正式编译,生成目标代码。
#define f(a)
(a)* b
若有:f(x+y) 则应有:
(x+y)b
若有:f(x+y+z) 则应有:
预编译后, z = f(x+y) 变成: z = x+y*b 结果为: z=31
(x+y+z)b
(这个结果与初始设想不一致)
“带参数宏定义” 必须注意以下几个方面: 1. 宏定义时要考虑实参(替换)的各种可能, 防止出 现二义性。
3. #include后的文件名既可用“ ”,也可用< >, 二者区别:
“ ”首先在当前目录中找,然后再去标准目录中找。
< > 只在标准目录(include目录)中找。
为提高预处理的搜索效率,通常对自定义的 非标准头文件使用方式一;而对系统提供的标准 头文件(如:math.h、stdio.h等)使用方式二。
将返回值 6 将返回值 8
但二者还是有区别的:
1) 宏替换在预编译时进行;
而函数调用在程序运行时进行
2) 宏展开时,仅仅是将宏体中的形参简单 地置换为实参,不计算实参值,也不会带来任何 返回值; 而函数调用要进行: l 计算实参值(假定用 2+3、 9–1作为实参将 被计算出来)、 l参数传递(将 5、8 传给形参x、y)、
这些文件要用到公用信息时,只要在文件 中加入#include “f.h”这么一行命令既可。这样 就不必在f1.c、…… 、fn.c每个文件中都去重 复定义这些公用的信息。
第9章_发酵液预处理和固液分离
36
板框过滤机动画
3)真空转鼓过滤机
主体是一个由筛板组成能转动的水平圆筒, 表面有一层金属丝网,网上覆盖滤布,圆筒内沿 径向被筋板分隔成若干个空间。
38
转筒真空过滤机动画
转筒真空过滤机
主要适用霉菌发酵液,对菌体 细小、黏度大铺助滤剂。
对于滤饼阻力较大的物料适应 能力较差。
40
带式真空过滤机
49
2)平抛式离心机
平抛式离心机一类结构简单的实验室常用的低 中速离心机,转速一般在 3000-6000rpm。
50
3)管式离心机
• 管式离心机具有一 个细长而高速旋转 的转鼓,转鼓内装 有纵向平板,
• 其下部有进料口。 上部两侧有重液相 和轻液相出口。
51
用于分离各种浮浊 液,进行液-液分离, 如油脂;
Al3+ >Fe3+ >H+ >Ca2+ >Mg2+ >K+ >Na+ >Li+
12
常用的凝聚剂电解质有:
硫酸铝 Al2(SO4)3•18H2O(明矾); 氯化铝 AlCl3•6H2O; 三氯化铁 FeCl3; 硫酸亚铁 FeSO4· 7H2O ; 石灰;ZnSO4;MgCO3
13
絮凝
flocculation
20
2.杂蛋白去除-变性
加热
大幅度调节pH值 加酒精、丙酮等有机溶剂或表面活性剂等。
21
3.杂蛋白去除- 吸附法
加入某些吸附剂或沉淀剂吸附杂蛋白质而除去。
四环类抗生素生产中,采用黄血盐和硫酸锌的 协同作用生成亚铁氰化锌钾K2Zn3[Fe(CN)5]2的 胶状沉淀来吸附蛋白质; 在枯草杆菌发酵液中,常加入氯化钙和磷酸氢 二钠形成沉淀物,该沉淀物不仅能吸附杂蛋白 和菌体等胶状悬浮物,加快了过滤速度。
第九章预处理命令(5503)
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 )。
9发酵产物的提取与纯化
密度/(kg/m3) 适用
干 湿
240~280 7.5~10
吸水率
备注
硅藻 土 珍珠 岩
130~190
250%
110~150
240~260
7.5
常用于抗 生素等提 取
• (二)细胞分离及破碎 • 固液分离操作常用方过程才能进一步进行产物 的提取分离,细胞分离往往是生物分离 的辅助步骤。 • 细胞破碎手段主要有机械法、酶法、物 理法和化学法等四大类。
• (二)萃取分离技术
• 溶剂萃取 • 双水相萃取 • 超临界萃取
• 1、溶剂萃取
• 利用欲提取的组分在溶剂中与原料液中 溶解度的差异来实现分离目的。
• 单级萃取 • 多级萃取
• 2、双水相萃取
• 用于蛋白质和核酸等生物大分子提取 • 常用的双水相有:PEG-葡聚糖和PEG -无机盐(磷酸盐、硫酸盐)。 • 双水相不仅可用于澄清发酵液处理,而 且还可以从含有菌体的原发酵液或细胞 匀浆中直接提取蛋白质。
• 3、使用助滤剂 • 发酵液中的胶体微粒被吸附到助滤剂微 粒上,而后者是不可压缩的,因而使压 缩性高的微生物菌体形成的滤饼可压缩 性大大降低,故可大大提高过滤速率。 • 常用助滤剂有硅藻土、珍珠岩等,对于 单细胞蛋白生产以及产物不能含硅的场 合,可使用淀粉作助滤剂。
常用助滤剂物化特性及助滤性能
• 3、盐析沉淀
利用各种生物分子在浓盐溶液中溶解度 的差异,通过向溶液中引入一定数量的 中性盐,使目的物或杂蛋白以沉淀析出, 达到纯化的目的。 • 优点:经济,不需特殊设备,操作简单, 安全,较少引起变性。通常蛋白质的分 离采用此法。
• 对于蛋白质的盐析沉淀,盐类选择原则为:
• (1)阴离子的沉淀效果顺序为: 柠檬酸根离子>PO33->SO42->CH3COO->Cl->NO3• (2)阳离子的沉淀效应顺序为NH4+>K+>Na+ • (3)加入固态盐类,以减少溶液的稀释。 • (4)加入盐的浓度应根据蛋白质大概浓度采用实 验或经验公式确定。
c程序设计第二版谭浩强课后答案
c程序设计第二版谭浩强课后答案C程序设计第二版是谭浩强教授编写的一本广泛使用的计算机程序设计教材,它以C语言为基础,深入浅出地介绍了程序设计的基本概念、语法规则和编程技巧。
这本书的课后习题对于加深理解C语言的知识点非常有帮助。
以下是部分课后习题的答案,供参考:第一章程序设计和C语言概述1. 问题一:简述程序设计的基本步骤。
- 答案:程序设计的基本步骤包括需求分析、设计、编码、测试和维护。
2. 问题二:C语言的主要特点是什么?- 答案:C语言的主要特点包括简洁高效、结构化、可移植性、丰富的运算符、灵活的数据类型和内存管理能力。
第二章 C语言程序的结构1. 问题一:C语言程序的基本结构是什么?- 答案:C语言程序的基本结构包括预处理指令、函数定义和主函数。
2. 问题二:什么是函数?C语言中函数的定义规则是什么?- 答案:函数是一段具有特定功能的代码块,可以被重复调用。
C 语言中函数的定义规则包括返回类型、函数名和参数列表。
第三章数据类型、运算符和表达式1. 问题一:C语言中的基本数据类型有哪些?- 答案:C语言中的基本数据类型包括整型(int)、字符型(char)、浮点型(float和double)。
2. 问题二:算术运算符有哪些?它们的优先级是怎样的?- 答案:算术运算符包括加(+)、减(-)、乘(*)、除(/)和模(%)。
它们的优先级从高到低依次是乘除、模、加减。
第四章控制语句1. 问题一:C语言中的条件语句有哪些?- 答案:C语言中的条件语句包括if语句、if...else语句和switch语句。
2. 问题二:循环语句有哪些?它们的基本结构是什么?- 答案:C语言中的循环语句包括while循环、do...while循环和for循环。
它们的基本结构是初始化、条件判断和迭代。
第五章数组1. 问题一:什么是数组?数组的声明方式有哪些?- 答案:数组是相同数据类型元素的集合。
数组的声明方式包括在函数内部声明和全局声明。
C语言程序设计 第3版 第9章 编译预处理
#include "test2.c" static int sum(int n) {
int i,s=0; for(i=1;i<=n;i++)
s=s+fact(i); return s; }
static int fact(int n) {
C语言程序设计
第9章 编译预处理
第1讲:编译预处理基本形式
提纲
1.宏定义 2.文件包含 3.条件编译
1.宏定义
不带参数宏定义 带参数宏定义
格式:
#define 标识符 字符串
功能:
指定标识符代替一个较复杂的字符串。
注意说明:
(1)宏名一般习惯用大写字母,例如宏名PI。 (2)宏名用做代替一个字符串,不作语法检查。 (3)宏定义无需在末尾加“;” (4)宏定义的有效范围为#undef命令终止。 (5)在进行宏定义时,可以引用已定义的宏名。
char web[50]; int i=0; gets(web); while(web[i]!='\0') {
#if(R==1) if(web[i]>='A'&&web[i]<='Z') {web[i]=web[i]+32; i++;}
#else if(web[i]>='a'&&web[i]<='z') {web[i]=web[i]-32; i++;}
形式3:
#ifndef 标识符 程序段1
#else 程序段2
编译预处理题与答案
第九章编译预处理9.1 选择题【题9.1】以下叙述中不正确的是。
A)预处理命令行都必须以#号开始B)在程序中凡是以#号开始的语句行都是预处理命令行C)C程序在执行过程中对预处理命令行进行处理D)以下是正确的宏定义#define IBM_PC【题9.2】以下叙述中正确的是。
A)在程序的一行上可以出现多个有效的预处理命令行B)使用带参的宏时,参数的类型应与宏定义时的一致C)宏替换不占用运行时间,只占编译时间D)在以下定义中C R是称为“宏名”的标识符#define C R 045【题9.3】请读程序:#define ADD(x) x+xmain(){int m=1,n=2,k=3;int sum=ADD(m+n)*k;printf(“sum=%d”,sum);}上面程序的运行结果是。
A)sum=9 B)sum=10 C)sum=12 D)sum=18【题9.4】以下程序的运行结果是。
#define MIN(x,y) (x)<(y)?(x):(y)main(){int i=10,j=15,k;k=10*MIN(i,j);printf(“%d\n”,k);}A)10 B)15 C)100 D)150【题9.5】在宏定义#define PI 3.14159中,用宏名PI代替一个。
A)常量B)单精度数C)双精度数D)字符串【题9.6】以下程序的运行结果是。
#include <stdio.h>#define FUDGE(y) 2.84+y#define PR(a) printf(“%d”,(int)(a))#define PRINT1(a) PR(a); putchar(‘\n’)main(){int x=2;PRINT1(FUDGE(5)*x);}A)11 B)12 C)13 D)15【题9.7】以下有关宏替换的叙述不正确的是。
A)宏替换不占用运行时间B)宏名无类型C)宏替换只是字符替换D)宏名必须用大写字母表示【题9.8】C语言的编译系统对宏命令的处理是。
C语言910章
C语言教案第九章预处理命令1.C语言中以“#”号开头的行,均为“编译预处理”命令行2.宏作为一个预处理命令,只是简单的文本代换3.在字符串代换过程中,一定要注意不要添加任何字符4.不带参数的宏定义及使用,注意宏中括号的用法5.*当宏定义在一行中写不下,需换行时应在最后一个字符后加“\”6.带参数的宏定义及使用7.注意宏名一般使用大写字母,而参数则不限8.文件包含的两种方法(书上212页)9.*条件编译习题217页:2,3,5第十章指针1.地址和指针的概念,指针型变量的定义2.指针型变量的基类型3.取地址运算符& 和指针运算符*4.通过指针为变量赋值及通过指针找出变量的值的方法5.*为指针变量赋“空”值:p=NULL或p=’\0’ 或p=0(NULL为预定义符,ASCII码值为0)6.函数的形参为指针变量时的数据传递7.通过传送地址值,在被调函数中直接改变主调函数中变量的值8.*函数返回地址值9.指针变量取代数组作函数参数10.一维数组与指针11.一维数组中的几种等价形式:设有定义int a[10], *p=a;数组元素的地址——&a[i], a+i, p+i, &p[i]数组元素——a[i], *(a+i), *(p+i), p[i]12.移动指针,每次移动一个单元13.*指针比较,通常两个或多个指针指向同一目标时比较才有意义14.*数组元素的地址作实参15.*函数的指针形参与函数体中数组的区别16.二维数组与指针17.一个二维数组可看作由多个一维数组构成18.几种等价形式:设有定义int a[5][6], *p=a[0]; 则存在着关于变量和变量的地址间的如下等价形式二维数组元素的地址——&a[i][j], a[i]+j, *(a+i)+j, &a[0][0]+列数*i+j,a[0]+列数*i+j 二维数组元素——a[i][j], *(a[i]+j), *(*(a+i)+j), *(&a[0][0]+列数*i+j),(*(a+i))[j]19.指向指针的指针:int a[5][6], **p=a;20.指针数组:int a[5][6], *p[4]; 定义了具有四个元素的基本整型指针数组21.行指针:int a[5][6], (*p)[6]; 定义了一个指向具有六个元素的一维数组的指向指针的指针型变量22.*二维数组名和指针数组作为实参在函数中的数据传递23.*二级指针和行指针作为实参在函数中的数据传递24.*字符串与指针,直接用字符型指针变量记录字符串的地址25.*指向函数的指针习题278页:2,3,4,9,10,14,15。
苏小红c语言程序设计课后答案
苏小红c语言程序设计课后答案苏小红的《C语言程序设计》是一本广泛使用的教材,它以清晰的结构和丰富的示例,帮助学生掌握C语言的基础知识和编程技巧。
课后答案对于学生来说是一个重要的学习资源,可以帮助他们检查自己的学习成果,加深对知识点的理解。
以下是一些可能的课后答案示例,用于帮助学生复习和理解C语言程序设计的相关概念。
第一章:C语言概述1. C语言的发展历史:C语言由Dennis Ritchie在20世纪70年代初期开发,最初用于UNIX操作系统的编写。
2. C语言的特点:C语言是一种结构化语言,具有高度的灵活性和强大的功能,能够进行底层系统编程。
第二章:C语言基础1. 数据类型:C语言提供了多种数据类型,包括整型(int)、浮点型(float和double)、字符型(char)等。
2. 变量声明:变量在使用前必须声明,声明时需要指定数据类型和变量名。
第三章:运算符和表达式1. 算术运算符:包括加(+)、减(-)、乘(*)、除(/)等。
2. 赋值运算符:如`=`,`+=`,`-=`等,用于给变量赋值或进行运算后赋值。
第四章:控制结构1. 条件语句:如`if`,`else if`,`else`,用于根据不同的条件执行不同的代码块。
2. 循环语句:包括`for`循环、`while`循环和`do-while`循环,用于重复执行代码块。
第五章:数组1. 一维数组:存储相同类型的元素,可以通过下标访问数组元素。
2. 多维数组:如二维数组,可以看作是数组的数组。
第六章:函数1. 函数定义:使用`return`类型和函数名来定义函数。
2. 函数调用:通过函数名和必要的参数来调用函数。
第七章:指针1. 指针变量:存储另一个变量的内存地址。
2. 指针运算:包括地址运算和指针的增减。
第八章:结构体和联合体1. 结构体:可以包含不同类型的数据成员。
2. 联合体:所有成员共享相同的内存位置。
第九章:预处理指令1. 宏定义:使用`#define`来定义常量或代码片段。
第九章改 预处理命令习题答案
第九章习题答案一、单项选择题1.A2.B3.C4.D5.B6.C7.A8.D9.D10.C11.B12.C13.D14.C二、填充题1.编译处理编译预处理2.非静态存储类型变量和外部函数3.74.printf(“%d\n”,m);5.fopen(“a.txt”,”rw”);6.x[i]>=’A’&&x[i]<=’Z’7.“ i=%d\n”8.(1) #define MAX(a,b,c)(2) #define MIN(a,b) (a<b?a:b)(3) #define isalnum(c) (c>=’0’&& c<=’9’)(4) #define isupper( c) (c>=’A’&& c<=’Z’)(5) #define islower( c) (c>=’a’ && c<=’z’)三、程序分析题1.运行结果: -32.运行结果: 7,473.运行结果:50 254.运行结果:x=9, y=55.运行结果:9 9 116.输出结果: x|y&z=3x^y&~z=1x&y&z=0!x|x=1~x|x=-1四、程序设计题1.解:#include <stdio.h>#include <math.h>#define S(a, b, c) 0.5* (a+b+c)#define AREA(a, b, c) sqrt (S(a, b, c)*(S(a, b, c)-a)* (S(a, b, c )-b)* (S(a, b, c)-c)) void main ( ){ float a, b, c;printf (“输入三角形的三条边长:a, b, c\n”);scanf (“%f, %f, %f”, &a, &b, &c) ;if (( a+b>c)&& (b+c>a) && (c+a>b)){ printf (“周长=%f\n”, 2*S(a, b, c ));printf (“面积=%f\n”, AREA(a, b, c ));}else printf (“a, b, c 的长度不能构成三角形\n”) ;}2.解:#include <stdio.h>#include <stdlib.h>void main (int argc, char *argv[ ]){ int a, b;if (argc<3) { printf (“Parameter missing!\n”) ; exit(0); }a=atoi (argv[1]) ;b=atoi (argv[2]) ; //在stdlib.h中有库函数atoi, 把字符串转换成整数printf (“%5d + %5d = %5d\n”, a, b, a+b) ;printf (“%5d - %5d = %5d\n”, a, b, a-b) ;printf (“%5d * %5d = %5d\n”, a, b, a*b) ;printf (“%5d / %5d = %5d\n”, a, b, a/b) ;printf (“%5d %% %5d = %5d\n”, a, b, a%b) ;}3.解:#include <stdio.h>#include <stdlib.h>void main (int argc, char *argv[]){ int i, sum=0;for (i=1; i<argc; i++) sum=sum+atoi (argv[i]) ;printf (“和= %d\n”, sum);}4.解:#include <stdio.h>#include <stdlib.h>#include <string.h>void main (int argc, char *argv[]){ if (argc<3){ printf (“Parameter missing!\n”) ; exit(0); } if ( strcmp (argv[1], argv[2])>0) puts (argv[1]) ;else puts (argv[2]) ;}。
谭浩强C程序设计(第三版)清华课件第9章_预处理
•
16、提出一个问题往往比解决一个更 重要。 因为解 决问题 也许仅 是一个 数学上 或实验 上的技 能而已 ,而提 出新的 问题, 却需要 有创造 性的想 像力, 而且标 志着科 学的真 正进步 。2021/6/29202 1/6/29J une
•
17、儿童是中心,教育的措施便围绕 他们而 组织起 来。202 1/6/292 021/6/2 92021/6/29202 1/6/29
printf("l=%10.4f\ns=%10.4f\nv=%10.4f\n",l,s,v); }
8
运行情况如下:
input radius: 4↙
说明:
1=25.1328 s=50.2655 v=150.7966
(1) 宏名Biblioteka 般习惯用大写字母表示,以便与变量名相 区别。但这并非规定,也可用小写字母。
• 经过预处理后程序可由编译程序对预处理后的 源程序进行通常的编译处理,得到可供执行的 目标代码。
4
基本概念
• C语言与其他高级语言的一个重要区别是可以 使用预处理命令和具有预处理的功能。
C提供的预处理功能主要有以下三种:
1.宏定义
2.文件包含
3.条件编译
这些功能分别用宏定义命令、文件包含命令 、条件编译命令来实现。为了与一般C语句相 区别,这些命令以符号“#”开头。例如: #define #include
• 这个标识符(名字)称为“宏名”。
• 在预编译时将宏名替换成字符串的过程称为“宏展 开”。#define是宏定义命令。
6
•
9、要学生做的事,教职员躬亲共做; 要学生 学的知 识,教 职员躬 亲共学 ;要学 生守的 规则, 教职员 躬亲共 守。20 21/6/29 2021/6/29Tues day , June 29, 2021
第九章 预处理命令
第九章 预处理命令一、选择题1.以下叙述不正确的是 。
A)预处理命令行都必须以#开始B)在程序中凡是以#开始的语句行都是预处理命令行C)C程序在执行过程中对预处理命令行进行处理D)预处理命令行可以出现在C程序中任意一行上2.以下叙述中正确的是 。
A)在程序的一行上可以出现多个有效的预处理命令行B)使用带参数的宏时,参数的类型应与宏定义时的一致C)宏替换不占用运行时间,只占用编译时间D)C语言的编译预处理就是对源程序进行初步的语法检查3.以下有关宏替换的叙述不正确的是 。
A)宏替换不占用运行时间B)宏名无类型C)宏替换只是字符替换D)宏名必须用大写字母表示4.在“文件包含”预处理命令形式中,当#include后面的文件名用””(双引号)括起时,寻找被包含文件的方式是 。
A)直接按系统设定的标准方式搜索目录B)先在源程序所在目录中搜索,再按系统设定的标准方式搜索C)仅仅搜索源程序所在目录D)仅仅搜索当前目录5.在“文件包含”预处理命令形式中,当#include后名的文件名用<>(尖括号)括起时,寻找被包含文件的方式是 。
A)直接按系统设定的标准方式搜索目录B)先在源程序所在目录中搜索,再按系统设定的标准方式搜索C)仅仅搜索源程序所在目录D)仅仅搜索当前目录6.在宏定义#define PI 3.1415926中,用宏名PI代替一个 。
A)单精度数B)双精度数C)常量D)字符串7.以下程序的运行结果是 。
#include<stdio.h>#define ADD(x) x+xvoid main ( ){ int m=1,n=2,k=3,sum ;sum = ADD(m+n)*k ;printf(“%d\n”,sum) ;A)9 B)10 C)12 D)188.以下程序的运行结果是 。
#include<stdio.h>#define MIN(x,y) (x)>(y) ? (x) : (y)void main ( ){ int i=10, j=15 , k;k = 10*MIN(i,j);printf(“%d\n”,k);}A)10 B)15 C)100 D)1509.以下程序的运行结果是 。
预处理命令ppt
12
第9章 预处理命令
说明 1、正因为带参宏定义本质还是简单字符替换(除了参数 正因为带参宏定义本质还是简单字符替换( 替换),所以容易发生错误。 ),所以容易发生错误 替换),所以容易发生错误。 不再是求 :#define 例:#define S(r) 3.14*r*r 面积了。 面积了。
宏展开
13
第9章 预处理命令
说明 3、带参数的宏定义与函数的本质区别 1)函数调用时,先求表达式的值,然后将值传递给形参; 函数调用时,先求表达式的值,然后将值传递给形参; 带参宏展开只在编译时进行的简单字符置换。 带参宏展开只在编译时进行的简单字符置换。 2)函数调用是在程序运行时处理的,给形参分配临时的 函数调用是在程序运行时处理的, 内存单元;而宏展开是在编译时进行的, 内存单元;而宏展开是在编译时进行的,展开时并不给 形参分配内存,也不进行“值传递”,也没有“返回值” 形参分配内存,也不进行“值传递” 也没有“返回值” 的概念。 的概念。 3)函数的形参要定义类型,且要求形参、实参类型一致。 函数的形参要定义类型,且要求形参、实参类型一致。 宏不存在参数类型问题。 宏不存在参数类型问题。 4)宏占用的是编译时间,函数调用占用的是运行时间。 宏占用的是编译时间,函数调用占用的是运行时间。 在多次调用时,宏使得程序变长,而函数调用不明显。 在多次调用时,宏使得程序变长,而函数调用不明显。
#define 编译预处理 #include 编译 连接 执行 宏定义 文件包含 条件编译
3
第9章 预处理命令
二、宏定义
(一)不带参数的宏定义
(二)带参数宏定义
4
第9章 预处理命令
(一)不带参数的宏定义(简单替换) 不带参数的宏定义(简单替换)
C语言程序设计 16-枚举、预处理等汇总
24
带参的宏定义使用举例
#define S(x,y) x*y main() { int a=3,b=4,c=5,d=6;
printf("a*b=%d\n", S (aa*b, b ) ) ; printf("a+b*c+d=%d\n" , S(a((+ab+)b*()c,(+cd+)d))));; }
enum weekday day; (2) enum weekday {Sun,Mon,Tue} day; (3) enum {Sun,Mon,Tue} day;
12
三、枚举型变量的使用
1)可对枚举变量赋值,但只能把枚举元素名赋给 枚举变量,不能把枚举元素值直接赋给枚举变量。 如: enum weekday {Sun,Mon, Tue, Wed, Thu,
10
一、枚举类型的定义
• 枚举元素的值也可在定义时重指定。例如: enum weekday {Sun=7,Mon=1, Tue, Wed, Thu,
Fri, Sat} ; 对于没有指定值的元素,按顺序加1,故: Tue是2,Wed是3,…,Sat是6。
11
二、枚举型变量的定义 枚举型变量的定义也有三种方式,如: (1) enum weekday { Sun,Mon,Tue };
3)可用枚举变量进行判断或比较操作。 enum flag {true,false} my_flag; … if (my_flag == true)
二级c语言第九章预处理命令
如果在程序中有下面的语句: R1=Radium(100); 那么,在编译预处理时,宏展开的顺序为从左到右进行置换,如果字符串 中包含宏中的形参,则将其用程序语句中相应的实参来替代,而字符串中的 其它字符原样保留。 因此,在处理上述语句时,将用 sqrt(area/PI) 替代 Radium(100),同时将 字符串sqrt(area/PI)中的形参area用实参l00来替代,并把已定义的宏PI的值代 入,经宏展开后,该语句变为“ R1=sqrt(100/3.14159); ”。
内蒙古科技大学 工程训练中心
预处理命令概述
所谓预处理,就是指源程序被正式编译之前所进行的处理工作,这 是C语言和其他高级语言之间的一个重要区别。
所有的预处理指令均以“#”开头,在它前面不能出现空格以外的字 符,而且在行结尾处没有分号。 “预处理命令” 的作用不是实现程序的功能,它们是发布给编译系 统的信息。它们告诉编译系统,在对源程序进行编译之前应该做些什么, 所以称这类语句为编译预处理命令。 C语言在执行一个C程序时,如果程序中有预处理命令,则首先进行 编译预处理(即根据预处理命令对源程序进行预先处理),然后再将经过 预处理的源程序编译成目标文件,而后进行目标文件的连接,当这些工 作都顺利通过后,才最终执行目标代码。这种执行过程大大提高了编程 效率。
PI 的有效范围
因为#undef 的作用是终止宏定义,因此PI的作用域从它定义开始到 #undef结束。在这以后,如果程序中出现 PI,则它不代表3.14159 。使 用#undef可以灵活控制宏定义的作用范围。
2019年1月8日5时29分 工程训练中心 徐国海
(二)带参数宏定义 (了解) 定义的一般格式为: #define 宏名(形式参数列表) 字符串
C语言9
第 九 章 编 译 预 处 理
7-1
宏定义 不带参数的宏定义 带参数的宏定义
7-1-1 7-1-2
7-2
7-3
文件包含
条件编译
9-1
宏定义
例9-1 请看下面的程序: #define PI 3.14159 main() {float r,l,s; scanf("%f",&r); l=2*PI*r; s=PI*r*r; printf("l=%7.2f,s=%7.2f\n",l,s); } 程序的第一行内容:#define PI 3.14159 就是一个宏定义,PI 称为宏名,其作用是用标识符PI代表3.14159这样一串字符。在此之后 的程序中就可以用标识符PI代替3.14159,称为宏引用。 预处理时,程序中的PI 将被所定义的串3.14159替换,称为宏展开 或宏替换,如程序中的语句:l=2*PI*r;和s=PI*r*r;中的宏展开后分别 为:l=2*3.14159*r;和s=3.14159*r*r;。 有两种形式的宏 :不带参数的宏和带参数的宏。
9-2
文件包含(1)
一个C 程序由若干个源程序组成,而一个源文件还可以将另一个源 文件的全部内容包含进来,即将指定源文件的内容包含在当前文件中。 例如有两个源文件file1.c和file2.c,文件的内容分别如下: file1.c的内容: int max(int x,int y) {int z; if(x>y) z=x; else z=y; return z; } file2.c的内容: #include "file1.c" main() {int a,b,c; scan("%d,%d",&a,&b); c=max(a,b); printf("c=%d\n",c); }
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
对于宏定义说明以下几点: 对于宏定义说明以下几点:
1. 宏定义是用宏名来表示一个字符串,在宏展 宏定义是用宏名来表示一个字符串, 开时又以该字符串取代宏名, 开时又以该字符串取代宏名,这只是一种简单 的代换,字符串中可以含任何字符, 的代换,字符串中可以含任何字符,可以是常 也可以是表达式, 数,也可以是表达式,预处理程序对它不作任 何检查。如有错误, 何检查。如有错误,只能在编译已被宏展开后 的源程序时发现。 的源程序时发现。 2. 宏定义不是说明或语句,在行末不必加分号, 宏定义不是说明或语句,在行末不必加分号, 如加上分号则连分号也一起置换。 如加上分号则连分号也一起置换。
预处理功能
9、1 、 宏定义
9、2 文件包含 、 9、3 条件编译 、
9.1
宏定义
在C语言源程序中允许用一个标识符来表 示一个字符串,称为“ 被定义为“ 示一个字符串,称为“宏”。被定义为“宏” 的标识符称为“宏名” 在编译预处理时, 的标识符称为“宏名”。在编译预处理时,对 程序中所有出现的“宏名” 程序中所有出现的“宏名”,都用宏定义中的 字符串去代换, 这称为“宏代换” 字符串去代换, 这称为“宏代换”或 宏展开” “宏展开”。 宏定义是由源程序中的宏定义命令完成的。 宏定义是由源程序中的宏定义命令完成的。 宏代换是由预处理程序自动完成的。在C语言 宏代换是由预处理程序自动完成的。 分为有参数和无参数两种。 中,“宏”分为有参数和无参数两种。
无参宏定义
1、格式:#define 标识符 字符串 其中宏名为C语言的标识符,一般用大写的字 母表示。“字符串”可以是常数、表达式、格 式串等。在前面介绍过的符号常量的定义就是 一种无参宏定义。此外,常对程序中反复使用 的表达式进行宏定义。例如: # define M (y*y+3*y) 2、功能:用于将一个指定的标识符替换为一个字 符串。
第九章 编译预处理
在源程序中放在函数之外, 而且一般都放在源文 件的前面,它们称为预处理部分。 所谓预处理是指在进行编译的第一遍扫描(词法扫 描和语法分析)之前所作的工作。预处理是C语言的一 个重要功能, 它由预处理程序负责完成。当对一个源 文件进行编译时, 系统将自动引用预处理程序对源程 序中的预处理部分作处理,处理完毕自动进入对源程 序的编译。 预处理命令不是C语言本身的组成部分,不能直接对 它们进行编译。 注意:预处理命令以“#”开头;
7. 可用宏定义表示数据类型,使书写方便。 可用宏定义表示数据类型,使书写方便。 例如: 在程序中可用STU作变 例如: #define STU struct stu在程序中可用 在程序中可用 作变 量说明: 量说明: STU body[5],*p; 应注意用宏定义表示数据类型和用typedef定义数据 应注意用宏定义表示数据类型和用 定义数据 说明符的区别。宏定义只是简单的字符串代换, 说明符的区别。宏定义只是简单的字符串代换,是在预 处理完成的, 是在编译时处理的, 处理完成的,而typedef是在编译时处理的,它不是作简 是在编译时处理的 单的代换,而是对类型说明符重新命名。 单的代换,而是对类型说明符重新命名。被命名的标识 符具有类型定义说明的功能。请看下面的例子: 符具有类型定义说明的功能。请看下面的例子: #define PIN1 int* 与 typedef (int*) PIN2;从形式上看这两者 从形式上看这两者 相似, 但在实际使用中却不相同。下面用PIN1,PIN2说 相似, 但在实际使用中却不相同。下面用 , 说 明变量时就可以看出它们的区别: 明变量时就可以看出它们的区别: PIN1 a,b;在宏代换后变成 int *a,b;表示 是指向 表示a是指向 在宏代换后变成 表示 整型的指针变量, 是整型变量。 整型的指针变量,而b是整型变量。然而:PIN2 a,b;表 是整型变量 然而: 表 示a,b都是指向整型的指针变量。因为PIN2是一个类型说 都是指向整型的指针变量。因为 是一个类型说 都是指向整型的指针变量 明符。由这个例子可见,宏定义虽然也可表示数据类型, 明符。由这个例子可见,宏定义虽然也可表示数据类型, 但毕竟是作字符代换。在使用时要分外小心,以避出错。 但毕竟是作字符代换。在使用时要分外小心,以避出错。
4、带参数的宏替换与函数有些相似,但有不同: 、带参数的宏替换与函数有些相似,但有不同: 宏替换产生的程序代码比使用函数时长。 1)、宏替换产生的程序代码比使用函数时长。 宏替换不需要进行参数传递、保护现场、 2)、宏替换不需要进行参数传递、保护现场、等操 作其执行速度比函数快。 作其执行速度比函数快。 3)、宏替换不存在返回值类型和参数类型的问题。 宏替换不存在返回值类型和参数类型的问题。 4)、使用宏时应注意: ) 使用宏时应注意: 定义宏时,最好对参数和宏体用括号括起来。 A、定义宏时,最好对参数和宏体用括号括起来。 如:#define square(n) n*n s=aquare(n+1 调用时 s=aquare(n+1) 则变成了s=a+1*a+1与预期效果不同。 则变成了s=a+1*a+1与预期效果不同。 s=a+ 定义带参数的宏时, B、定义带参数的宏时,宏名和左括号之间不能有空 否则被认为是无参数的宏. 格,否则被认为是无参数的宏. C、对宏的定义 只是进行字符的替换 不进行语法检 只是进行字符的替换,不进行语法检 、对宏的定义,只是进行字符的替换 查.
3. 宏定义必须写在函数之外,其作用域为宏定 义命令起到源程序结 束。如要终止其作用域可 使用# undef命令,例如: # define PI 3.14159 main() { …… } # undef PIPI的作用域 f1() 表示PI只在main函数中有效,在f1中无效。
4. 宏名在源程序中若用引号括起来,则预处理程 宏名在源程序中若用引号括起来, 序不对其作宏代换。 序不对其作宏代换。intf("OK"); printf("\n"); } 上例中定义宏名OK表示 表示100,但在 上例中定义宏名 表示 ,但在printf语 语 句中OK被引号括起来,因此不作宏代换。程序的 被引号括起来, 句中 被引号括起来 因此不作宏代换。 运行结果为: 这表示把 这表示把“ 当字符串处理。 运行结果为:OK这表示把“OK”当字符串处理。 当字符串处理
6. 宏定义也可用来定义多个语句,在宏调用时,把 宏定义也可用来定义多个语句,在宏调用时, 这些语句又代换到源程序内。看下面的例子。 这些语句又代换到源程序内。看下面的例子。 #define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h; main() { int l=3,w=4,h=5,sa,sb,sc,vv; SSSV(sa,sb,sc,vv); printf("sa=%d\nsb=%d\nsc=%d\nvv=%d\n",sa,s b,sc,vv); } 程序第一行为宏定义,用宏名SSSV表示 个赋值语 表示4个赋值语 程序第一行为宏定义,用宏名 表示 个形参分别为4个赋值符左部的变量 个赋值符左部的变量。 句,4 个形参分别为 个赋值符左部的变量。在宏调 用时, 个语句展开并用实参代替形参。 用时,把4 个语句展开并用实参代替形参。使计算结 果送入实参之中。 果送入实参之中。
5. 宏定义允许嵌套,在宏定义的字符串中 宏定义允许嵌套, 可以使用已经定义的宏名。 可以使用已经定义的宏名。在宏展开时由 预处理程序层层代换。例如: 预处理程序层层代换。例如: #define PI 3.1415926 #define S PI*y*y /* PI是已定义的 PI是已定义的 宏名*/对语句 对语句: 宏名 对语句: printf("%f",s);在宏代换后 在宏代换后 变为: 变为: printf("%f",3.1415926*y*y); 6. 习惯上宏名用大写字母表示,以便于与 习惯上宏名用大写字母表示, 变量区别。但也允许用小写字母。 变量区别。但也允许用小写字母。
4. 在宏定义中,字符串内的形参通常要用括号括 在宏定义中, 起来以避免出错。 起来以避免出错。 #define SQ(y) y*y 代换成: 代换成:sq=a+1*a+1; 注意:对于宏定义不仅应在参数两侧加括号, 注意:对于宏定义不仅应在参数两侧加括号, 也 应在整个字符串外加括号。 应在整个字符串外加括号。 5. 带参的宏和带参函数很相似,本质上不同, 带参的宏和带参函数很相似,本质上不同,
对于带参的宏定义说明: 对于带参的宏定义说明:
1. 带参宏定义中,宏名和形参表之间不能有空格出现。 带参宏定义中,宏名和形参表之间不能有空格出现。 例如把: 写为: 例如把: #define MAX(a,b) (a>b)?a:b 写为: #define MAX (a,b) (a>b)?a:b 将被认为是无参宏定义, 将被认为是无参宏定义, 宏名MAX代表字符串 (a,b) (a>b)?a:b。宏展开时,宏调用 宏名 代表字符串 。宏展开时, 语句: 将变为: 语句: max=MAX(x,y);将变为: 将变为 max=(a,b)(a>b)?a:b(x,y);这显然是错误的。 这显然是错误的。 这显然是错误的 2. 在带参宏定义中,形式参数不分配内存单元,因此不必作 在带参宏定义中,形式参数不分配内存单元, 类型定义。而宏调用中的实参有具体的值。 类型定义。而宏调用中的实参有具体的值。要用它们去代 换形参,因此必须作类型说明。 换形参,因此必须作类型说明。这是与函数中的情况不同 在函数中,形参和实参是两个不同的量, 的。在函数中,形参和实参是两个不同的量,各有自己的 作用域,调用时要把实参值赋予形参,进行“值传递” 作用域,调用时要把实参值赋予形参,进行“值传递”。 而在带参宏中,只是符号代换,不存在值传递的问题。 而在带参宏中,只是符号代换,不存在值传递的问题。
二、带参数的宏替换
1、定义:#define 宏名(形式参数表) 宏体 、定义: 宏名(形式参数表) 2、引用:宏名(实际参数表) 、引用:宏名(实际参数表) 3、对带参数的宏,不仅对宏名进行替换,而且对参数 、对带参数的宏,不仅对宏名进行替换, 进行替换。 进行替换。 例如: <stdio. 例如:#include <stdio.h> #define PI 3.14 #define circuit(r) 2*PI*r #define area(r) PI*r*r main() { float ra,c,a; ra,c,a; scanf(“% ,&ra) ,&ra); scanf( %f”,&ra); c=circuit(ra); c=circuit(ra); a=area(ra); a=area(ra); printf(“ra= f,c=%f,a=% ,ra,c,a) ra=% ,ra,c,a); printf( ra=%f,c=%f,a=%f”,ra,c,a);}