赞
踩
所有发生的事情都一定有迹可循,如果问心无愧,就不需要掩盖也就没有迹象了,如果问心有愧,
就必然需要掩盖,那就一定会有迹象,迹象越多就越容易顺藤而上,这就是推理的途径。顺着这条途径顺流而下就是犯罪,逆流而上,就是真相。一名优秀的程序员是一名出色的侦探。每一次调试都是尝试破案的过程。
我们是如何写代码的?
又是如何排查出现的问题的呢?
调试(英语:Debugging / Debug),又称除错,是发现和减少计算机程序或电子仪器设备中程序错误的一个过程。
①发现程序错误的存在
②以隔离、消除等方式对错误进行定位
③确定错误产生的原因
④提出纠正错误的解决办法
⑤对程序错误予以改正,重新测试
Debug 通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序。
Release 称为发布版本,它往往是进行了各种优化,使得程序在代码大小和运行速度上都是最优
的,以便用户很好地使用。
- int main()
- {
- int arr[10] = { 0 };
- int sz = sizeof(arr) / sizeof(arr[0]);
- for (int i = 0; i < sz; i++)
- {
- arr[i] = i + 1;
- }
- return 0;
- }
Debug版本:可以调试,较大,研发人员写的
Relseae版本:不可以调试,较小,测试人员测试的
Debug版本和Release版本运行上可能有一定的差异,这是因为Release版本在运行时做了一些优化,这些优化可能导致结果的差异
在环境中选择 debug 选项,才能使代码正常调试
F5 | 启动调试,经常用来直接跳到下一个断点处 |
F9 | 创建断点和取消断点 |
F10 | 逐过程,通常用来处理一个过程,一个过程可以是一次函数调用,或者是一条语句 |
F11 | 逐语句,就是每次都执行一条语句,但是这个快捷键可以使我们的执行逻辑进入函数内部 |
Ctrl+F5 | 开始执行不调试,如果你想让程序直接运行起来而不调试就可以直接使用 |
3.3.1 查看临时变量的值
3.3.2 查看内存信息
3.3.3 查看调用堆栈:反馈函数的调用逻辑
3.3.4 查看汇编信息
(1)第一种方式:右击鼠标,选择【转到反汇编】
(2)第二种方式
3.3.5 查看寄存器信息
实例1:实现代码:求 1!+2!+3! ...+ n! ;不考虑溢出
- int main()
- {
- int i = 0;
- int sum = 0;//保存最终结果
- int n = 0;
- int ret = 1;//保存n的阶乘
- scanf_s("%d", &n);
- for (i = 1; i <= n; i++)
- {
- int j = 0;
- for (j = 1; j <= i; j++)
- {
- ret *= j;
- }
-
- sum += ret;
- }
- printf("%d\n", sum);
- return 0;
- }
输入:5
输出:34863
可以发现,此时代码存在运行错误
1. 首先推测问题出现的原因。初步确定问题可能的原因最好
2. 实际上手调试很有必要
3. 调试的时候我们心里有数
- int main()
- {
- int i = 0;
- int sum = 0;//保存最终结果
- int n = 0;
- int ret = 1;//保存n的阶乘
- scanf_s("%d", &n);
- for (i = 1; i <= n; i++)
- {
- int j = 0;
- for (j = 1; j <= i; j++)
- {
- ret *= j;
- }
- sum += ret;
- //修改
- ret = 1;
- }
- printf("%d\n", sum);
- return 0;
- }
输入:5
输出:153
实例2:
- int main()
- {
- int i = 0;
- int arr[10] = { 0 };
- for (i = 0; i <= 12; i++)
- {
- arr[i] = 0;
- printf("hehe\n");
- }
- return 0;
- }
输出:无限个hehe循环
这是因为数组越界导致在修改arr的值的时候,同时修改掉了i的值
i和arr是局部变量,局部变量是放在内存中的栈区的,栈区内存的使用习惯是先使用高地址空间,再使用低地址空间
而数组随着下标的增长,地址是由低变高变化的
环境 | 两个变量之间间隔的整形数 |
VC6.0 | 0 |
gcc | 1 |
vs13-19 | 2 |
实例3:在Linux x86-64 gcc环境下,下面的程序会出现什么问题?运行结果是什么?为什么?
- int main(int argc,char* argv[])
- {
- long i;
- long a[16];
- for (int i = 0; i <= 17; i++)
- {
- a[i] = 0;
- printf("%d", i);
- }
- return 0;
- }
结果:死循环
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。