第一篇:C语言的预处理命令有哪些?
问:关于C语言中的预处理命令? 答:
我们可以在C源程序中插入传给编译程序的各种指令,这些指令被称为预处理器指令(等价于汇编语言中的伪指令),它们扩充了程序设计的环境。
现把常用的预处理命令总结如下:
1.预处理程序
按照ANSI标准的定义,预处理程序应该处理以下12条指令:
#if、#ifdef、#ifndef、#else、#elif、#endif、#define、#undef、#line、#error、#pragma、#include。
显然,所有的12个预处理指令都以符号#开始,每条预处理指令必须独占一行。2.#define #define指令定义一个标识符和一个串(也就是字符集),在源程序中发现该标识符时,都用该串替换之(原样替换,不要附加任何人为理解上的符号)。这种标识符称为宏名字,相应的替换称为宏代换。一般形式如下:
#define macro-name char-sequence 这种语句不用分号结尾。宏名字和串之间可以有多个空格符,但串开始后只能以新行终止。在C语言里宏定义只用来做的宏名替换,而不做语法检查的,因而它不是C语句,所以宏定义的语句结尾不需要加分号。
宏也在C里也叫预处理命令,因为宏是在程序编译前先进行字符替换的,所以叫预处理.例如:我们使用LEFT代表1,用RIGHT代表0,我们使用两个#define指令:
#define LEFT 1 #define RIGHT 0 每当在源程序中遇到LEFT或RIGHT时,编译程序都用1或0替换。定义一个宏名字之后,可以在其他宏定义中使用,例如:
#define ONE 1 #define TWO ONE+ONE #define THREE ONE+TWO 宏代换就是用相关的串替代标识符。因此,如果希望定义一条标准错误信息时,可以如下定义:
#define ERROR_MS “Standard error on input n” 如果一个串长于一行,可在行尾用反斜线””续行,如下:
#define LONG_STRING “This is a very very long
string that is used as an example”
3.#error
#error指令强制编译程序停止编译,它主要用于程序调试(放在错误的分支中,一旦进入错误的分支就显示该信息)。
#error指令的一般形式是:
#error error-message 注意,宏串error-message不用双引号包围。遇到#error指令时,错误信息被显示,可能同时还显示编译程序作者预先定义的其他内容。
4.#include 程序中的#include指令要求编译程序读入另一个源文件。被读入文件的名字必须用双引号(“”)或一对尖括号(<>)包围,例如:
#include “stdio.h” #include
包含文件中可以包含其他#include指令,称为嵌套包含。允许的最大嵌套深度随编译器而变。文件名被双括号或尖括号包围决定了对指定文件的搜索方式。文件名被尖括号包围时,搜索按编译程序作者的定义进行,一般用于搜索某些专门放置包含文件的特殊目录。当文件名被双引号包围时,搜索按编译程序实时的规定进行,一般搜索当前目录。如未发现,再按尖括号包围时的办法重新搜索一次。
通常,绝大多数程序员使用尖括号包围标准的头文件,双引号用于包围与当前程序相关的文件名。
5.条件编译指令
若干编译指令允许程序员有选择的编译程序源代码的不同部分,这种过程称为条件编译。5.1 #if、#else、#elif #endif 条件编译指令中最常用的或许是#if,#else,#elif和#endif。这些指令允许程序员根据常数表达式的结果有条件地启用(包围)部分代码。
#if的一般形式是:
#if constant-expression Statement sequence #endif 如#if后的常数表达式为真,则#if和#endif中间的代码被编译,否则忽略该代码段。#endif标记#if块的结束。
#else指令的作用与C语言的else相似,#if指令失败时它可以作为备选指令。例如:
#include
#if MAX>99 printf(“Compiled for array greater than 99.n”);#else printf(“Complied for small array.n”);#endif return 0;} 注意,#else既是标记#if块的结束,也标记#else块的开始。因为每个#if只能写一个#endif匹配。#elif指令的意思是“否则,如果”,为多重编译选择建立一条if-else-if(如果-否则-如果链)。如果#if表达式为真,该代码块被编译,不测试其他#elif表达式。否则,序列中的下一块被测试,如果成功则编译之。一般形式如下:
#if expression 1 Statement sequence 1 #elif expression 2 Statement sequence 2 #elif expression 3 Statement sequence 3 …
#elif expression n Statement sequence n #endif 5.2 #ifdef 和 #ifndef
条件编译的另一个方法是使用编译指令#ifdef和#ifndef,分别表示“如果已定义”和“如果未定义”。#ifdef的一般形式如下:
#ifdef macro-name Statement sequence #endif 如果macro-name原先已经被一个#define语句定义,则编译其中的代码块。#ifndef的一般形式是:
#ifndef macro-name Statement sequence #endif 如果macro-name当前未被#define语句定义,则编译其中的代码块。
我认为,用这种方法可以很方便的开启/关闭整个程序的某项特定功能。
?? #ifdef和#ifndef都可以使用#else或#elif语句。
#inlucde
#undef指令删除前面定义的宏名字。也就是说,它的意思是“不要已定义的某个宏”。一般形式为:
#undef macro-name 7.使用defined 除#ifdef之外,还有另外一种确定是否定义宏名字的方法,即可以将#if指令与defined编译时操作符一起使用。defined操作符的一般形式如下:
defined macro-name 如果macro-name是当前定义的,则表达式为真,否则为假。例如,确定宏MY是否定义,可以使用下列两种预处理命令之一:
#if defined MY 或
#ifdef MY 也可以在defined之前加上感叹号”!”来反转相应的条件。例如,只有在DEBUG未定义的情况下才编译。
#if!defined DEBUG Printf(“Final Version!n”);#endif 使用defined的一个原因是,它允许由#elif语句确定的宏名字存在。??? 8.#line #line指令改变__LINE__和__FILE__的内容。__LINE__和__FILE__都是编译程序中预定义的标识符(见11)。标识符__LINE__的内容是当前被编译代码行的行号,__FILE__的内容是当前被编译源文件的文件名。#line的一般形式是:
#line number “filename”
其中,number是正整数并变成__LINE__的新值;可选的“filename”是合法文件标识符并变成__FILE__的新值。#line主要用于调试和特殊应用。
9.#pragma
#pragma是编译程序实现时定义的指令,它允许由此向编译程序传入各种指令。例如,一个编译程序可能具有支持跟踪程序执行的选项,此时可以用#pragma语句选择该功能。编译程序忽略其不支持的#pragma选项。#pragma提高C源程序对编译程序的可移植性。
10.预处理操作符#和## 有两个预处理操作符:#和##,它们可以在#define中使用。
操作符#通常称为字符串化的操作符,它把其后的串变成用双引号包围的串。例如: #include
Printf(“I like C”);
操作符##把两个标记拼在一起,形成一个新标记。例如: #include
Printf(“%d”,xy);
操作符#和##主要作用是允许预处理程序对付某些特殊情况,多数程序中并不需要。11.预定义宏
C规范了5个固有的预定义宏,它们是: __LINE__:正在编译的程序的行号 __FILE__:正在编译的程序的文件名
__DATE__:代表源文件翻译成目标码的日期,形如month/day/year(月/日/年)__TIME__:代表源代码编译成目标码的时间,形如hour:minute:second(时:分:秒)__STDC__:如果__STDC__的内容是十进制常数1,则表示编译程序的实现符合标准C。
问:在程序的一行上可以出现多个有效的预处理命令行。预处理命令可以出现在函数的内部。给出的两段代码,有一段是错误的,哪段???
答:
第1段有问题。#ifndef WIN32 #endif printf(“OKn”);在这里,这个printf就不会被执行。也就是说,一行中,只能有一条预处理指令,当编译的预处理阶段,编译器识别了一条完整的预处理指令后,后面的所有东西他都不要了。
对于第二段,在函数里,我们是可以使用预处理指令的。比如:
void fun(void){ #ifdef WIN32...// 对于windows系统环境的操作 #else...// 对于windows以外的系统环境的操作
#endif /* WIN32 */...}
问:两个C语言的小问题(预处理命令)1.#define abc(x,y)(x)<(y)?(x):(y)main(){ int a=10,b=15,c;c=10*abc(a,b);printf(“%d”,c);} 我算出结果是100,可参考答案是15,不知是为什么? 2.#define A 5.5 #define B(x)A*x*x main(){ int a=1,b=2;;printf(“%f”,B(a+b));} 这个答案是9.5,这又是为何?
答:
1.c = 10*abc(a,b)= 10 *(x)<(y)?(x):(y)= 10 * 10 < 15 ? 10 : 15 = 100 < 15 ? 10 : 15 = 15 2.B(a+b)= A * a+b * a+b = 5.5 * 1 + 2 * 1 + 2 = 5.5 + 2 + 2 = 9.5
问:C语言预处理命令#define 选择题:以下程序的运行结果是?
#define MAX(A,B)(A)>(B)?(A):(B)#define PRINT(Y)printf(“Y=%dt”,Y)main(){ int a=1,b=2,c=3,d=4,t;t=MAX(a+b,c+d);PRINT(t);} A)Y=3 B)存在语法错误 C)Y=7 D)Y=0 请给我解释下PRINT(t);在宏展开是怎么表示的,答案是C。
答:宏处理的时候,一定要记住:直接代进去,任何多余的动作都不能有(别想当然地加括号!)。
还有记住一点的是在printf“"双括号里的是不能替代的话,所以这里的Y是输出形式。而不是字符常量Y的替代。这是一种特殊规定。
PRINT(t)=printf(”Y=%dt“,t);所以结果必将是Y=(一个值)又因为
MAX(a+b,c+d)(a+b)>(c+d)?(a+b):(c+d)的结果是7,所以,答案是“C)Y=7”。
问:求解:关于c语言中,宏定义的问题。。#define NLMSG_ALIGNTO 4 #define NLMSG_ALIGN(len)(((len)+ NLMSG_ALIGNTO1))这句编译没错,请问 NLMSG_ALIGNN(len)的值是多少
答: 1.第一句定义了一个符号常量,值为4。隐含的作用是指定地址对齐方式:按4边界对齐。
例如若某个对象的长度为18,那么在为其分配空间时,通过NLMSG_ALIGN宏就可以计算出最接近其的4的倍数为((18+4-1)& ~(3))= 20,这样便为其申请/分配20字节空间。
这是32位微控制器/微处理器中,为了防止非对齐操作产生Exception(异常)而添加的保护措施。
2.第二句定义了一个宏,宏体是一个表达式
3.NLMSG_ALIGNN(len)是宏调用,具体做法就是单纯地把字符用符号常量的值替换掉(宏展开)。宏展开的结果就是(((len)+ 41))。要说一下 &~,&是与运算,~是取反,都是对二进制”位"的操作。这里&~在一起,意思是“与3的非”。比如 5&~3 那么结果就是4(用2进制数表达就是:101&~011那么就是101&100,结果是2进制的100)。
问: 答:
问:
答:
第二篇:【好程序员学习笔记】C语言 编译预处理命令
文件包含:
把指定的文件插入到预处理命令行所在的位置并取代该命令行,即把指定的文件和当前的源程序文件连接成一个源文件。#include<文件名>
在文件包含目录中去查找指定的文件,并将该文件添加到源文件中。一个被包含的文件中可以含有文件包含命令来包含另一个文件。#include“文件名”
命令中文件名的位置是当前源文件的位置,若在当前目录中未找到该文件,则再到“包含目录”中去查找。
宏
用一个标识符表示一个字符串,称为宏,被定义为宏的标识符称为宏名。在编译预处理时对程序中所有出现的宏名用宏定义中的字符串去代换,这就是宏替换。它是由系统编译程序时自动完成的。
无参宏定义
#define 标识符 字符串 如
#define PI 3.14 使用宏时要注意:
(1)宏定义是用宏名来表示一个字符串,在宏展开时用字符串取代宏名。
2)宏定义不是变量定义或语句,在行末不能加分号,如果加上分号则分号也成为字符串的一部分。
(3)宏定义可以出现在程序的任何地方,其作用域是宏定义命令所在位置开始到源程序结束。如果要终止其作用域可使用#undef命令。
(4)宏定义允许嵌套,在宏定义的字符串中可以使用已经定义的宏名。在宏展开时将逐层替换。
(5)#define PI 3.1415926 #define AREA PI*y*y 有参宏定义
#define 宏名(形参表)字符串
对带参数的宏,在调用中不仅要进行宏展开,而且还要用实参去替换形参。带参宏调用的语法格式如下: 宏名(实参表); #define M(x)x+1 K=M(3);K=3+1 定义有参宏时要注意以下几点:
(1)有参宏定义中,宏名和形参表之间不能有空格出现。
(2)在函数中,调用时要把实参的值赋给形参,进行“值传递”。而在带参宏调用中,只是符号替换,不存在值传递问题。
(3)宏定义中的形参只能是标识符,而宏调用中的实参可以是表达式。
字符串内的形参通常用括号括起来以避免出错。
在使用宏时不仅要将形式参数用括号括起来,还应将其整体用括号括起来。#define S(y)((y)*(y))条件编译
条件编译可以按不同的条件去编译不同的程序段。1#ifdef #ifdef 标识符 程序段1 #else 程序段2 #endif 功能:如果标识符已被#define定义过,则对程序段1进行编译,否则对程序段2进行编译。格式中的#else部分可以没有,即: #ifdef 标识符
程序段 #endif
2、#ifndef 语法格式如下: #ifndef 标识符 程序段1
#else 程序段2 #endif 功能与ifdef形式的功能正好相反。
3、#if 常量表达式 #if 常量表达式 程序段1 #else 程序段2 #endif 功能:如果常量表达式的值为真(非0),则对程序段1进行编译,否则对程序段2进行编译,因此可以使程序在不同条件下,完成不同的功能。
上面介绍的条件编译实现的逻辑也可以用条件语句来实现,但是用条件语句将会对整个源程序进行编译,生成的目标代码程序较长,运行时间也较长,采用条件编译则根据条件只编译其中的程序段1或程序段2,生成的目标程序较短。
第三篇:黑马程序员C语言教程: CC++培训专家-预处理命令使用详解
传智播客C/C++培训专家:预处
理命令详解
作为一枚C/C++程序猿,在我们编写和查看C/C++源代码的过程中会遇到各种编译指令,这些指令称为预处理命令。预处理命令虽然不是C/C+的一部分,但却扩展了C程序的设计环境,下面传智播客C/C+培训专家将向大家介绍如何应用预处理程序和注释简化程序开发过程,并提高程序的可读性。
ANSI标准定义的C语言所有预处理命令均以符号#开头,比如我写程序时常用的:
#define,#undef,#include,#if,#else,#elif,#endif,#ifdef,#ifndef, #error 1.#define和 #undef 宏定义命令的一般形式为:
#define[MacroName][MacroValue],示例如下: #defineITHEIMA 传智播客黑马程序员
在源程序中每次遇到ITHEIMA时,均以定义的值传智播客黑马程序员代换它。
在使用该宏时,有以下几点注意事项:
该语句没有分号。在标识符和串之间可以有任意个空格。 定义宏的时候,可以使用之前已经定义好的宏。
如果串长于一行,可以在该行末尾用一反斜杠' '续行。
#defineLONG_STRING“good good study,day day up!” 在定义宏标识符时,字母一般需要大写。 预处理运算符的使用:
#--该符号是“字符串化”的意思,出现在宏定义中的#是把跟在后面的参数转换成一个字符串
#define ERROR_LOG(module)
fprintf(stderr, “error: ”#module“n”)ERROR_LOG(“add”);转换为 fprintf(stderr,“error: ”add“n”);ERROR_LOG(devied =0);转换为 fprintf(stderr,“error: devied=0n”); ##--是连接符号,将多个串连接到一起。char *szStr = “传播播客_黑马程序员”;#define ITCAST(exp)cout < 2.#include 命令#i nclude使编译程序将另一源文件嵌入带有#include的源文件,被读入的源文件必须用双引号或尖括号括起来。例如: #include“stdio.h”或者#include 这两行代码均使用C编译程序读入并编译用于处理磁盘文件库的子程序。将文件嵌入#i nclude命令中的文件内是可行的,这种方 式称为嵌套的嵌入文件,嵌套层次依赖于具体实现。 如果显式路径名为文件标识符的一部分,则仅在那些子目录中搜索被嵌入文件。 例如: #include “../include/head.h” 如果文件名用双引号括起来,则首先检索当前工作目录。如果未发现文件,则在命令行中说明的所有目录中搜索。如果仍未发现文件,则搜索实现时定义的标准目录。例如: include “head.h” 如果文件名被尖括号括起来,则首先在编译命令行中的目录内检索。如果文件没找到,则检索标准目录,不检索当前工作目录。 例如: include
它与#endif之间的代码,否则跳过这些代码。命令#endif标识一个#if块的结束。
跟在#if后面的表达式在编译时求值,因此它必须仅含常量及已定义过的标识符,不可使用变量。表达式不许含有操作符sizeof(sizeof也是编译时求值)。
#else命令的功能有点象C语言中的else;#else建立另一选择(在#if失败的情况下)。注意,#else属于#if块。
#elif命令意义与ELSE IF 相同,它形成一个if else-if阶梯状语句,可进行多种编译选择。#elif 后跟一个常量表达式。如果表达式为true,则编译其后的代码块,不对其它#elif表达式进行测试。否则,顺序测试下一块。
4.#error 命令#error强迫编译程序停止编译,主要用于程序调试。该指令使预处理器发出一条错误消息,该消息包含指令中的文本.这条指令的目的就是在程序崩溃之前能够给出一定的信息。
5.#ifdef 和 #ifndef 条件编译的另一种方法是用#ifdef与#ifndef命令,它们分别表示“如果有定义”及“如果无定义”。# ifdef的一般形式是:
# ifdef macroname statement sequence
#endif #ifdef与#ifndef可以用于#if、#else,#elif语句中,但必须与一个#endif。
define MAX 91 #include
int main(){ #ifdef MAX cout<<“hello,MAX!”< #else cout<<“where is MAX?”< #endif #ifndef LEO cout<<“LEO is not defined”< #endif return 0;} 今天关于预处理命令的知识点传智播客C/C+培训专家就为大家接受到这里, 欢迎大家留言交流.
第四篇:铁水预处理
什么是铁水预处理?
铁水预处理是指铁水兑入炼钢炉之前,为脱硫或脱硅、脱磷而进行的处理过程。
除上述普通铁水预处理外还有特殊铁水预处理,如针对铁水含有特殊元素提纯精炼或资源综合利用而进行的提钒、提铌、提钨等预处理技术。
2、在炼钢生产中采用铁水预脱硫技术的必要性是什么?
(1)用户对钢的品种和质量要求提高,连铸技术的发展也要求钢中硫含量低(硫含量高容易使连铸坯产生裂纹)。铁水脱硫可满足冶炼低硫钢和超低硫钢种的要求。
(2)转炉炼钢整个过程是氧化气氛,脱硫效率仅为30%~40%;而铁水中的碳、硅等元素含量高,氧含量低,提高了铁水中硫的活度系数,故铁水脱硫效率高;铁水脱硫费用低于高炉、转炉和炉外精炼的脱硫费用。
(3)减轻高炉脱硫负担后,能实现低碱度、小渣量操作,有利于冶炼低硅生铁,使高炉稳定、顺行,可保证向炼钢供应精料。
(4)有效地提高钢铁企业铁、钢、材的综合经济效益。
3、铁水脱硫常用的脱硫剂有几类,各有何特点?
生产中,常用的脱硫剂有苏打灰(Na2C03)、石灰粉(CaO)、电石粉(CaC2)和金属镁等。以上脱硫剂可以单独使用,也可以几种配合使用。
(1)苏打灰。其主要成分为Na2C03,铁水中加入苏打灰后,与硫作用发生以下3个化学反应:
Na2C03(1)+[S]+2[C]=Na2S(1)+3{CO}
Na2C03(1)+[S]十[Si]=Na2S(1)+SiO2(S)+{CO}
Na20(1)+[S]=Na2S(1)+[O]
用苏打灰脱硫,工艺和设备简单,其缺点是脱硫过程中产生的渣会腐蚀处理罐的内衬,产生的烟尘污染环境,对人有害。目前很少使用。
(2)石灰粉。其主要成分为CaO,用石灰粉脱硫的反应式如下:
2CaO(S)+[S]+(1/2)[Si]=(CaS)(S)+(1/2)(Ca2Si04)
石灰价格便宜、使用安全,但在石灰粉颗粒表面易形成2CaO?Si02致密层,限制了脱硫反应进行,因此,石灰耗用量大,致使生成的渣量大和铁损大,铁水温降也较多。另外,石灰还有易吸潮变质的缺点。
(3)电石粉。其主要成分为CaC2,电石粉脱硫的反应式如下:
CaC2+[S]=(CaS)(S)+2[C] 用电石粉脱硫,铁水温度高时脱硫效率高,铁水温度低于1300℃时脱硫效率很低。另外,处理后的渣量大,且渣中含有未反应尽的电石颗粒,遇水易产生乙炔(C2H2)气体,故对脱硫渣的处理要求严格。在脱硫过程中也容易析出石墨碳污染环境。电石粉易吸潮生成乙炔(乙炔是可燃气体且易发生爆炸),故电石粉需要以惰性气体密封保存和运输。
(4)金属镁。镁喷入铁水后发生如下反应:
Mg+[S]=MgS(S)
镁在铁水的温度下与硫有极强的亲和力,特别是在低温下镁脱硫效率极高,脱硫过程可预测,硫含量可控制在0.001%的精度。这是其他脱硫剂所不能比拟的。
金属镁活性很高,极易氧化,是易燃易爆品,镁粒必须经表面钝化处理后才能安全地运输、储存和使用。钝化处理后,使其镁粒表面形成一层非活性的保护膜。
用镁脱硫,铁水的温降小,渣量及铁损均少且不损坏处理罐的内衬,也不影响环境。因而铁水包喷镁脱硫工艺获得了迅猛的发展。
镁的价格较高,保存时须防止吸潮。
4、铁水脱硫的主要方法有哪些,铁水脱硫技术的发展趋势是怎样的?
迄今为止,人们已开发出多种铁水脱硫的方法,其中主要方法有:投入脱硫法、铁水容器转动搅拌脱硫法、搅拌器转动搅拌脱硫法和喷吹脱硫法等。
(1)投入法。该法不需要特殊设备,操作简单,但脱硫效果不稳定,产生的烟气污染环境。
(2)铁水容器搅拌脱硫法。该法主要包括转鼓法和摇包法,均有好的脱硫效果,该法容器转动笨重,动力消耗高,包衬寿命低,使用较少。
(3)采用搅拌器的机械搅拌法。如KR法(见图3-1)即属于此类。
KR搅拌法由于搅拌能力强和脱硫前后能充分的扒渣,可将硫含量脱至很低,其缺点是设备复杂,铁水温降大。
(4)喷吹法。此法是用喷枪以惰性气体为载体,将脱硫剂与气体混合吹入铁水深部,以搅动铁水与脱硫剂充分混合的脱硫方法。该法可以在鱼雷罐车(混铁车)或铁水包内处理铁水。铁水包喷吹法目前已被广泛应用。图3-2为鱼雷罐车喷吹法脱硫装置示意图。
喷吹脱硫法具有脱硫反应速度快、效率高、操作灵活方便,处理铁水量大,设备投资少等优点。因而,它已成为铁水脱硫的主要方法。
铁水脱硫技术的发展趋势如下:
(1)采用全量铁水脱硫工艺;
(2)趋向在铁水包内预脱硫;
(3)脱硫方法以喷吹法为主;
(4)用金属镁做脱硫剂的趋势不断扩大。
5、用金属镁进行铁水脱硫的机理是什么?
镁为碱土金属,相对原子质量为24.305,密度为1.738g/cm3;熔点为651℃;沸点为1107℃。当金属镁与硫结合生成MgS后,其熔点为 2000℃,密度为2.8g/cm3,如与氧结合生成MgO后,其熔点为2800℃,密度为3.07~3.20g/cm3,二者均为高熔点、低密度稳定化合物。
镁通过喷枪喷入铁水中,镁在高温下发生液化、气化并溶于铁水:
Mg(S)→Mg(1)→{Mg}→[Mg]
Mg与S的相互反应存在两种情况:
第一种情况:{Mg}+[S]=MgS(S)
(3-1)
第二种情况:{Mg}→[Mg]
(3-2)
[Mg]+[S]=MgS(S)
(3-3)
在高温下,镁和硫有很强的亲和力,溶于铁水中的[Mg]和{Mg}都能与铁水中的[S]迅速
反应生成固态的MgS,上浮进入渣中。
在第一种情况下,在金属-镁蒸气泡界面,镁蒸气与铁水中的硫反应生成固态MgS,这只能去除铁水中3%~8%的硫。
在第二种情况下,溶解于铁水中的镁与硫反应生成固态MgS,这是主要的脱硫反应,最为合理。在这种情况下,保证了镁与硫的反应不仅仅局限在镁剂导入区域或喷吹区域内进行,而是在铁水包整个范围内进行,这对铁水脱硫是十分有利的。
镁在铁水中的溶解度取决于铁水温度和镁的蒸气压。镁的溶解度随着压力的增加而增大,随铁水温度的上升而大幅度降低。为了获得高脱硫效率,必须保证镁蒸气泡在铁水中完全溶解,避免未溶解完的镁蒸气逸人大气造成损失。促进镁蒸气大量溶解于铁水中的措施是:铁水温度低;加大喷枪插入铁水液面以下的深度,提高镁蒸气压力,延长镁蒸气泡与铁水接触时间。
6、采用金属镁脱硫为什么要对镁粒进行表面钝化处理,对颗粒镁有什么要求?
金属镁活性很高,极易氧化,是易燃易爆品。镁粒只有经表面钝化处理后才能安全地运输、储存和使用。经钝化处理后,镁粒表面形成一层非活性的保护膜,如盐钝化的涂层颗粒镁,制备时采用熔融液态镁离心重复分散技术,利用空气动力逆向冷却原理将盐液包敷在镁颗粒外层,形成银灰色均匀的球状颗粒。
单吹镁脱硫用的涂层颗粒镁要求:
wMg≥92%;粒度为0.5~1.6mm,其中粒度大于3mm以上的针状不规则颗粒少于8%。
7、铁水脱硫容器为什么趋向采用铁水包?
在鱼雷罐内进行脱硫,动力学条件较差,脱硫剂喷入后,由于鱼雷罐形状影响搅拌的均匀性,反应重现性差,脱硫剂消耗量大。采用铁水包喷吹脱硫,由于铁水包的几何形状,使脱硫反应具有更好的动力学条件和反应空间,可根据冶炼具体要求更准确地控制铁水的硫含量。一般容量大于80t的铁水包铁液深度都比鱼雷罐深,喷入铁水的脱硫剂与铁水进行反应更加充分,因此在铁水包内喷吹脱硫可以有效利用脱硫剂。同时铁水包内的铁水温度比鱼雷罐内低一些,更促进镁脱硫获得理想的脱硫效果,降低了铁水处理成本。由于铁水包内喷吹脱硫有较高的效率,与在鱼雷罐脱硫相比,如果将硫含量从0.045%降到0.010%,可节省脱硫剂 15%;如果将硫含量从0.045%降到0.005%,可节省脱硫剂24%。显然,硫含量的目标值越低,在铁水包喷吹脱硫剂的优势越大。20世纪80年代已开始发展到在铁水包内处理铁水。目前新建铁水脱硫装置大多采用铁水包单独喷吹镁或复合喷吹镁的技术和设备。
8、铁水包单吹颗粒镁脱硫的工艺流程及基本工艺参数是怎样的?
铁水包单吹颗粒镁脱硫工艺系统装备如图3-3所示。
单吹颗粒镁脱硫工艺流程如图3-4所示。
脱硫剂采用单一的颗粒金属镁,流动性好,喷吹罐配备了专门的计量给料装置。为保证把镁剂(不掺添加料)可靠地喷入铁水中并使镁的吸收率在95%以上,且不堵枪,应合理选择喷枪和输镁管路的结构和喷吹系统参数。应使供氮压力稳定,喷枪端面距包底约0.2m,喷枪结构要保证为镁溶解于铁水并继而被吸收创造良好的条件。喷枪浸入深度不足2.4m的铁水
包,喷枪端部要装备锥形气化室。整个脱硫过程可采用计算机自动控制,每次处理前只需输入初始硫含量、目标硫含量、铁水温度、铁水重量等参数,脱硫处理过程便可自动进行。
单吹颗粒镁脱硫工艺参数如下:
(1)脱硫剂颗粒镁,粒度为0.5~1.6mm,wMg≥92%
(2)氮气压力
1.0MPa
(3)初始铁水
w[sl=0.035%
(4)目标铁水
w[s]=0.005%
(5)喷吹时间
≤10min
(6)脱硫剂(Mg)流量
8~15kg/min
(7)脱硫剂(Mg)消耗
0.46kg/t
(8)温降
10℃
9、铁水包镁基复合喷吹脱硫的工艺流程及基本工艺参数是怎样的?
镁基脱硫剂是由镁粉加上石灰粉或电石粉及其他添加剂组成,喷入铁水后脱硫反应主要由镁粉完成。复合喷吹的镁粉和石灰粉(或电石粉)分别存贮在两个喷吹罐内,用载气输送,在管道内混合。通过调节分配器的粉料输送速度来确定两种粉料的比例,对镁粉流动性无要求。整个脱硫过程可采用计算机自动控制,每次处理前只需输人初始硫含量、目标硫含量、铁水温度、铁水重量等参数、脱硫处理过程便可自动进行。
镁基复合喷吹脱硫工艺流程如图3-5所示。镁基复合脱硫工艺参数如下:
(1)脱硫剂
Mg+CaO(2)氮气压力
1.1MPa(3)初始铁水w[s]
0.035%(4)目标铁水w[s]
0.005%
(5)喷吹时间
10min
(6)脱硫剂流量
Mg粉12kg/min
石灰粉45kg/min
(7)脱硫剂消耗
Mg粉0.65kg/t
石灰粉1.92kg/t
(8)温降
20℃
1O、喷镁脱硫要求铁水包净空是多少?
当铁水包喷镁脱硫时,镁通过喷枪喷入铁水,载气对铁水有搅拌作用,可以促进反应物的传质和产物的排出。由于镁在高温下液化、气化和溶于铁水,气化时产生的镁气体对铁水的搅拌作用强烈,顶吹时常发生喷溅。因此,铁水包应有不小于400mm高度的净空,同时设置防溅包盖是必要的。
11、铁水包喷吹镁脱硫与其他脱硫工艺比较具有哪些优点?
铁水包喷镁脱硫工艺与其他脱硫工艺相比,具有以下显著的优点:
(1)脱硫效率高。可根据冶炼品种要求,铁水硫含量可脱至任意水平,深脱硫时达到ws=0.005%以下,甚至wS=0.002%以下;
(2)脱硫剂单耗低,处理时间短;
(3)形成渣量少,扒渣铁损低;(4)对环境污染小;(5)温度损失少;
(6)易于进行过程自动控制;
(7)综合成本低。
第五篇:C语言程序设计教案 第九章 编译预处理
第九章 编译预处理
课题:
教学目的: 教学重点: 教学难点: 第九章 编译预处理
1、了解预处理的概念及特点
2、掌握有参宏与无参宏的定义及使用,领会文件包含的使用及效果 掌握宏的使用,文件包含 有参宏与无参宏的使用
步骤一 复习引导
ANSI C标准规定可以在C源程序中加入一些“预处理命令”,以改进程序设计环境,提高编程效率。
这些预处理命令是由ANSI C统一规定的,但它不是C语言本身的组成部分,不能直接对它们进行编译。必须在对程序进行通常的编译之前,先对程序中这些特殊的命令进行“预处理”,即根据预处理命令对程序作相应的处理。经过预处理后程序不再包括预处理命令了,最后再由编译程序对预处理后的源程序进行通常的编译处理,得到可供执行的目标代码。
步骤二 讲授新课
C语言与其他高级语言的一个重要区别是可以使用预处理命令和具有预处理的功能。C提供的预处理功能主要有以下三种: 宏定义、文件包含、条件编译。
分别用宏定义命令、文件包含命令、条件编译命令来实现。为了与一般C语句相区别,这些命令以符号“ #” 开头。
§9.1宏定义
宏: 代表一个字符串的标识符。
宏名:被定义为“宏”的标识符。
宏代换(展开):在编译预处理时,对程序中所有出现的 “宏名”,用宏定义中的字符串去代换的过程。
一、不带参数的宏定义
一般形式:
#define
标识符
字符串
#define PI 3.1415926
main()
{ 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.4n”, 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;
printf(“s=%dn”,s);} 先宏展开:s=3*(y*y+3*y)+4*(y*y+3*y)+ y*(y*y+3*y)再与源程序合并
说明:
⑴宏名一般用大写表示,以便与变量名区分。⑵使用宏名使程序易读,易修改。⑶只作简单的置换,不作正确性检查。⑷宏定义不是C语句,不必在行末加分号。
⑸宏名的作用域一般从自定义命令到本源文件结束。⑹可以用# undef命令终止宏定义的作用域。⑺宏定义允许嵌套,允许层层置换。
⑻宏名在源程序中用双引号括起来,则TC中预处理不对其作宏代换。
例: printf(“L=%f”, L);中双引号内L不替换。
⑼宏定义与定义的变量不同,宏只作字符替换,不分配内存空间。⑽对“输出格式”进行宏定义,可以减少书写麻烦 例如:
#define P printf #define D “%d,%d,%dn”
#define F “%6.2f,%6.2f,%6.2fn” main(){ int a=5,c=8,e=11;
float b=3.8,d=9.7;f=21.08;
P(D,a,c,e);
P(F,b,d,f);
P(F,a+b,c+d,e+f);}
二、带参数的宏定义
格式:#define
宏名(参数表)字符串
例:#define s(a,b)a*b
{……
area =s(3,2);
……}
对带参的宏展开后,为area=3*2;
例: #define M(y)y*y+3*y
{……
k=M(5);
……}
对其展开后,为k=5*5+3*5;
说明:
⑴对带参数的宏展开只是将宏名后括号内的实参
字符串代替#define命令行中的形参。
⑵宏定义时,在宏名与带参数的括号之间不应加
空格,否则将空格以后的字符都作为替代字符
串的一部分。
⑶带参宏定义,形参不分配内存单元,因此不必
作类型定义。(与函数的区别之一)⑷带参宏与函数的区别之二:
例如: main(){ int i=1;
while(i<=5)
printf(“%dt”,SQ(i++));} SQ(int y){ return(y)*(y);} 其结果为:1
如:
#define SQ(y)((y)*(y))main(){ int i =1;
while(i<=5)
printf(“%dt”,SQ(i++));} 运行结果: 2 12 30
例:利用宏定义求两个数中的大数。
#define MAX(a,b)(a>b)?a:b main(){int x, y, max;
scanf(“%d%d”, &x, &y);
max =MAX(x, y);
printf(“max=%dn”, max);} 带参的宏定义和函数不同:
1、函数调用时,先求实参表达式值,后代入。而带参的宏只是进行简单的字符替换。
2、函数调用是在程序运行时处理的,分配临时的内存单元。而宏展开则是在编译时进行的,不分配内存单元,不进行值的传递,也无“返回值”。
3、对函数中的实参和形参都要定义类型,类型应一致。而宏不存在类型问题,宏名和参数无类型,只是一个符号代表,展开时代入指定的字符即可。
4、调用函数只可得到一个返回值,而用宏可以设法得到几个结果。
5、使用宏次数多时,宏展开后使源程序增长,而函数调用不使源程序变长。
6、宏替换不占运行时间,只占编译时间。
而函数调用则占用运行时间(分配单元、保留现场、值传递、返回)一般用宏代表简短的表达式比较合适。
也可利用宏定义实现程序的简化。例9.5:
#define PR printf #define NL “n” #define D “%d” #define D1 D NL #define D2 D D NL #define D3 D D D NL #define D4 D D D D NL #define S “%s” main(){ int a,b,c,d;
char string[]=“CHINA”;
a=1;b=2;c=3;d=4;
PR(D1,a);
PR(D2,a,b);
PR(D3,a,b,c);
PR(D4,a,b,c,d);
PR(S, string);}
§9.2 “文件包含”处理
“文件包含”处理是指将指定的被包含文件的全部内容插到该控制行的位置处,使其成为源文件的一部分参与编译。因此,被包含的文件应该是源文件。通常置于源程序文件的首部,故也称为“头文件”。
C编译系统提供的头文件扩展名为“.h”,但设计者可根据实际情况,自行确定包含文件的后缀、名字及其位置。
一般形式,#include “文件名”
或
#include <文件名>
文件format.h
#define PR printf #define NL “n” #define D “%d” #define D1 D NL #define D2 D D NL #define D3 D D D NL #define D4 D D D D NL #define S “%s” 文件file1.c #include “format.h” main(){ int a,b,c,d;
char string[]=“CHINA”;
a=1;b=2;c=3;d=4;
PR(D1,a);
PR(D2,a,b);
PR(D3,a,b,c);
PR(D4,a,b,c,d);
PR(S, string);} 注:被包含的文件应是源文件,而不应是目标文件。
头文件除了可以包含函数原型和宏定义外,也可以包括结构体类型定义和全局变量定义等。说明:
1、一个include命令只能指定一个被包含文件,如果要包含n个文件,要用n个include命令。
2、如果文件1包含文件2,而文件2中要用到文件3的内容,则可在文件1中用两个include命令分别包含文件2和文件3,而且文件3应出现在文件2之前,即在“file1.c”中定义:
#include “file3.h”
#include “file2.h”
3、在一个被包含文件中又可以包含另一个被包含文件,即文件包含是可以嵌套的。
4、被包含文件(file2.h)与其所在的文件(file1.c),在预编译后已成为同一个文件。
5、在#include 命令中,文件名可以用双引号或尖括号括起来。
如: #include
或
#include “file2.h” 二者的区别:
用尖括号时称为标准方式,系统到存放C库头文件所在的目录中寻找要包含的文件。
用双引号时,系统先在用户当前目录中寻找要包含的文件,若找不到,再按标准方式查找。#include “c:tcincludemyfile.h”
/*正确*/ #include
/*正确*/ #include
/*错误*/
用尖括号:带路径:按指定路径去寻找被包含文件,但此时被包含文件不能以“.h”结尾,否则错误。
不带路径:仅从指定标准目录下找。
用引号: 带路径:按指定路径去寻找被包含文件,不再从当前目录和指定目录下找。
不带路径:先在当前目录下找,找不到再在系统指定的标准目录下找。
步骤三 课堂小结
本课主要讲解了宏定义、“文件包含”处理。对带参数的宏的使用,及与函数的使用的区别。搞清经常使用的头文件。
步骤四 布置作业 上机作业:(第九章课后练习)9.4 书面作业:(第九章课后练习)9.7、9.8