c程序编译流程
程序的基本流程如图:
1. 预处理
预处理相当于根据预处理指令组装新的C/C++程序。经过预处理,会产生一个没有宏定义,没有条件编译指令,没有特殊符号的输出文件,这个文件的含义同原本的文件无异,只是内容上有所不同。
读取C/C++源程序,对其中的伪指令(以#开头的指令)进行处理
①将所有的“#define”删除,并且展开所有的宏定义
②处理所有的条件编译指令,如:“#if”、“#ifdef”、“#elif”、“#else”、“endif”等。这些伪指令的引入使得程序员可以通过定义不同的宏来决定编译程序对哪些代码进行处理。预编译程序将根据有关的文件,将那些不必要的代码过滤掉。
③处理“#include”预编译指令,将被包含的文件插入到该预编译指令的位置。
(注意:这个过程可能是递归进行的,也就是说被包含的文件可能还包含其他文件)
删除所有的注释
-
添加行号和文件名标识。
以便于编译时编译器产生调试用的行号信息及用于编译时产生的编译错误或警告时能够显示行号
保留所有的#pragma编译器指令
2. 编译
将预处理完的文件进行一系列词法分析、语法分析、语义分析及优化后,产生相应的汇编代码文件。
3. 汇编
将编译完的汇编代码文件翻译成机器指令,并生成可重定位目标程序的.o文件,该文件为二进制文件,字节编码是机器指令。
汇编器是将汇编代码转变成机器可以执行的指令,每一个汇编语句几乎都对应一条机器指令。所以汇编器的汇编过程相对于编译器来讲比较简单,它没有复杂的语法,也没有语义,也不需要做指令优化,只是根据汇编指令和机器指令的对照表一一翻译即可。
4. 链接
通过链接器将一个个目标文件(或许还会有库文件)链接在一起生成一个完整的可执行程序。
由汇编程序生成的目标文件并不能立即就被执行,其中可能还有许多没有解决的问题。
例如,某个源文件中的函数可能引用了另一个源文件中定义的某个符号(如变量或者函数调用等);在程序中可能调用了某个库文件中的函数,等等。所有的这些问题,都需要经链接程序的处理方能得以解决。
链接程序的主要工作就是将有关的目标文件彼此相连接,也就是将在一个文件中引用的符号同该符号在另外一个文件中的定义连接起来,使得所有的这些目标文件成为一个能够被操作系统装入执行的统一整体。
命令行下编译文件
因为VS自带了开发人员选项,所以我们不需要再去配置环境变量了(感谢微软~)。
- 在win10中询问Cortana,输入VS,选择VS2015开发人员命令提示。
- 进入.c或者.cpp文件所在路径,输入命令:cl *.cpp,编译预先用记事本或者Editplus写好的文件,查看结果
- 如上图所示,生成了两个文件,一个输出文件:hello.exe,一个obj文件:hello.obj;查看文件夹:
- 执行.exe文件,查看运行结果,如下图:
说明
在以上的编译过程中我们只用了cl的编译命令就帮我们最终的可执行文件HelloWorld.exe,这是因为cl.exe程序在编译时自己会去调用link.exe、lib.exe等程序。
可通过”cl -help “查看常用的编译选项
选项 | 作用 |
---|---|
/O1 | 创建小代码 |
/O2 | 创建快速代码 |
/Oa | 假设没有别名 |
/Ob | 控制内联展开 |
/Od | 禁用优化 |
/Og | 使用全局优化 |
/Oi | 生成内部函数 |
更详细的中文介绍也可参考这篇博文:
http://www.lellansin.com/%E5%BE%AE%E8%BD%AF-cl-exe-%E7%BC%96%E8%AF%91%E5%99%A8.html
静态库
打开VS2015,新建一个项目,选择win32项目,点击确定,选择静态库这个选项,预编译头文件不选。
在这个空项目中,添加一个.h文件和4个.cpp文件。名字我们起为head.h和add.cpp,div.cpp,mul.cpp,sub.cpp
head.h文件:
#include"head.h"
int add(int a, int b)
{
return a + b;
}
div.cpp:
#include"head.h"
double div(int a, int b)
{
return (a + 0.0) / (b + 0.0);
}
mul.cpp:
#include"head.h"
int mul(int a,int b)
{
return a*b;
}
sub.cpp:
#include"head.h"
int sub(int a, int b)
{
return a - b;
}
编译这个项目之后(生成——》生成解决方案),会在debug文件夹下生成mymath.lib文件,这个就是我们需要的静态链接库。
下面说明如何调用静态链接库。
- 首先需要新建一个空项目,起名为test_mymath。
- 将之前static项目下的head.h这个文件复制到mymath项目的目录下
- 在工程中加入head.h文件,快捷键:Ctrl + Alt +A添加现有项:
新建一个20145211hzy.cpp文件如下:
编译运行可得结果:
#pragma comment(lib,"mymath.lib"),这一句是显式的导入静态链接库。除此之外,还有其他的方法,比如通过设置路径等等,这里不做介绍。
或者 选中工程名,点击右键 -> 属性, 在“ VC++目录 ”的包含目录里面加入head.h的所在目录的路径,库目录里面加入mymath.lib所在目录的路径;在链接器->输入->附加依赖项输入“mymath.lib”。然后源文件里添加#include"head.h"就OK了。
动态库
- 创建DLL
#ifdef DLL_API
#else
#define DLL_API _declspec(dllimport)
#endif
DLL_API int add(int a, int b);
#include "dll.h"
#define DLL_API _declspec(dllexport)
int add(int a, int b)
{
return a + b;
}
double div(int a, int b)
{
return (a + 0.0) / (b + 0.0);
}
int mul(int a,int b)
{
return a*b;
}
int sub(int a, int b)
{
return a - b;
}
6.编译
- 调用DLL
1.在该解决方案中新建一个win32控制台项目;
2.在新项目上单击右键,设为启动项;
3.再次在新项目上单击右键->添加->引用->项目->解决方案,将DLL项目前边的方框选中。
4.添加代码:
#include "stdafx.h"
_declspec(dllexport) int add(int a, int b);
_declspec(dllexport) int sub(int a, int b);
_declspec(dllexport) int MUL(int a, int b);
_declspec(dllexport) double div(int a, int b);
int main()
{
int a=3, b=4;
printf("%d\n", add(a,b));
printf("%d\n", sub(a,b));
printf("%lf\n", divs(a, b));
printf("%d\n", mul(a, b));
system("pause");
return 0;
}
5.编译完成。
- 其他
如果创建与调用不在同一个解决方案内,需要将生成的mymath.lib文件复制到win32项目文件夹中,并添加引用命令:
#pragma comment(lib,"mymath.lib"),或者在项目->属性->VC++目录中手动添加;
再将mymath.dll文件放入win32项目的调试文件夹中。
关于调试
调试是VS2015很强大的一项功能,它可以帮我们找到程序中的很多错误。
开始调试
先贴出我的测试代码(需要说明的是,这段代码并没有问题):
#include <stdio.h>
#include"windows.h"
int sum(int m);
int main()
{
int * pi;
int i, n = 0;
pi = &i;
int arr[3] = { 10, 20, 30 };
pi = arr;
sum(50);
for (i = 0; i <= 50; i++)
{
n += i;
}
printf("The sum of 1-50 is %d \n", n);
system("pause");
}
int sum(int m)
{
int i, n = 0;
for (i = 1; i <= m; i++)
{
n += i;
}
printf("The sum of 1-50 is %d \n", n);
return n;
}
1. 设置断点:可以直接在你需要设置断点的地方按F9,或者在那行右键->断点->插入断点(下面会介绍如何设置函数、临时、条件断点);双击红点,撤销断点。
2. 调试:逐过程调试:按F10,或者点击菜单栏的调试–>逐过程;
逐语句调试:按F11,或者点击菜单栏的调试–>逐语句
函数断点:
- 在main处设置断点,逐过程调试,因为在main处设置了断点,main函数还未执行
- 继续按F10,直到运行完毕
两个断点间的调试(关于函数断点的逐过程和逐语句):
在调试过程中F5是执行到下一个断点。F10是逐过程,与逐语句不同的是,在执行到下图中断点时,再执行会执行断点下面的语句,而不是去执行语句中的方法。
而F11是逐语句,在执行到下图中的断点 时,按F11会执行到sum里面逐步记录执行过程。
条件断点:
设置条件断点非常容易。在特定的行上,按F9设置断点;然后右击断点–编辑窗口左侧的红点,在上下文菜单上选择“条件”。
现在我再运行这个程序按F5达到条件断点,只有当x等于5时,程序运行才会被中断。对于其它条件下的x值,断点将被跳过。
3. 完毕:
Alt+F4强行退出调试:
3. 相关快捷键总结:
F6: 生成解决方案
Ctrl+F6: 生成当前项目
F7: 查看代码
Shift+F7: 查看窗体设计器
F5: 启动调试
Ctrl+F5: 开始执行(不调试)
Alt+F4: 停止调试
Ctrl+Shift+F5: 重启调试
F9: 切换断点
Ctrl+F9: 启用/停止断点
Ctrl+Shift+F9: 删除全部断点
F10: 逐过程
Ctrl+F10: 运行到光标处
F11: 逐语句
总结
VS2015是一个非常好的软件,我今天分享的知识冰山一角,可以做的还有很多!