c语言的程序环境与预处理详解
作者:.SacaJawea
c语言代码的实现包含两种环境
1.翻译环境,将源代码转化成可执行的机器指令
2.执行环境,执行代码
1.翻译环境
包括两个过程,编译与链接
·程序中每一个源文件通过编译器转化成目标文件(obj)
·这些目标文件又通过链接器捆绑在一起
·链接器同时会链接标准库中的函数以及程序员个人的库到程序中
·形成可执行代
编译分成三个阶段:预编译(预处理)->编译->汇编
预处理阶段
1.完成头文件的引用
2.#define定义的宏与符号的替换
3.注释的删除
编译阶段:将代码转化成汇编代码
1.语法分析
2.词法分析
3.语义分析
4.符号汇总
汇编阶段:生成符号表,将汇编指令转化成二进制的机器指令
链接阶段:将多个目标文件与链接库进行链接
1.合并段表
2.符号表的重定向与重定位
2.运行环境
1.在有操作系统的环境中,程序由操作系统加载到内存中。在独立的环境中,这个操作手动完成。
2.程序从main函数开始执行
3.程序调用堆栈
4.终止程序
3.预处理详解
3.1#define定义的符号
符号名一般大写,后面不要加上;
#define MAX 100
3.2#define定义的宏
#define SQUARE(x) ((x) * (x))
宏的名字与参数之间不能加空格;尽量多加括号,避免错误
3.3#define的替换规则
#define M 100 #define MAX(X,Y) (X) > (Y) ? (X) : (Y) int main() { int max = MAX(M,50);//这里的M会被先替换成100 printf("%d",max); }
1.如果有宏,先查看宏的参数有没有#define定义的符号,如果有则替换
2.将替换的文本与程序中的宏替换3.再查看是否有定义的符号,如果有则替换
替换#define定义的符号时,不会替换字符串常量中的内容。
宏的参数中能出现define定义的符号,但是宏不能实现递归
3.4#与##
在宏的定义中(# + 参数)能将参数转换成对应的字符串
##能连接两个符号
4.宏与函数对比
宏的优点
1.宏的参数类型不用声明
2.当运算量较小时,宏的运算时间与代码量是远小于函数的
缺点
1.宏类型无关,不够严谨
2.宏不能调试
3.每次使用宏的时候,一份宏定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序的长度。
4.宏可能带来运算优先级的问题
5.#undef
用于移除宏定义
6.条件编译
#if与#endif为一组,如果#if后的表达式为真,执行代码直到#endif
但是#if后面不能写变量
多个分支的条件编译#if,#elif,#elif,#else,#endif
判断是否被定义
#if defined(symbol)
#ifdef symbol
#if !defined(symbol)
#ifndef symbol
7.文件包含
#include后用双引号,编译器会先从本地目录下查找,找不到再去库函数中找。用<>编译器直接在库函数中找
为了防止头文件被反复包含,用#pragma once,写在头文件第一行
或者用
#ifndef NAME
#define NAME
//头文件内容
#endif
总结
到此这篇关于c语言的程序环境与预处理详解的文章就介绍到这了,更多相关c语言程序环境与预处理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!