当前位置:   article > 正文

调试技巧-C语言_c语言可以查看运行过的代码吗

c语言可以查看运行过的代码吗

1.什么是bug

第一次被发现的导致计算机错误的飞蛾,也是第一个计算机程序错误。

2.调试

所有发生的事情都一定有迹可循,如果问心无愧,就不需要掩盖也就没有迹象了,如果问心有愧,
就必然需要掩盖,那就一定会有迹象,迹象越多就越容易顺藤而上,这就是推理的途径。顺着这条途径顺流而下就是犯罪,逆流而上,就是真相。一名优秀的程序员是一名出色的侦探。每一次调试都是尝试破案的过程。

我们是如何写代码的?

 又是如何排查出现的问题的呢?

2.1 什么是调试 

调试(英语:Debugging / Debug),又称除错,是发现和减少计算机程序或电子仪器设备中程序错误的一个过程。

2.2 调试的基本步骤

①发现程序错误的存在
②以隔离、消除等方式对错误进行定位
③确定错误产生的原因
④提出纠正错误的解决办法
⑤对程序错误予以改正,重新测试

2.3 Debug和Release的介绍

Debug 通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序。

Release 称为发布版本,它往往是进行了各种优化,使得程序在代码大小和运行速度上都是最优
的,以便用户很好地使用。

  1. int main()
  2. {
  3. int arr[10] = { 0 };
  4. int sz = sizeof(arr) / sizeof(arr[0]);
  5. for (int i = 0; i < sz; i++)
  6. {
  7. arr[i] = i + 1;
  8. }
  9. return 0;
  10. }

Debug版本:可以调试,较大,研发人员写的

Relseae版本:不可以调试,较小,测试人员测试的

Debug版本和Release版本运行上可能有一定的差异,这是因为Release版本在运行时做了一些优化,这些优化可能导致结果的差异

3. Windows环境调试介绍

3.1 调试环境的准备

在环境中选择 debug 选项,才能使代码正常调试

3.2 快捷键

F5启动调试,经常用来直接跳到下一个断点处
F9创建断点和取消断点
F10逐过程,通常用来处理一个过程,一个过程可以是一次函数调用,或者是一条语句
F11逐语句,就是每次都执行一条语句,但是这个快捷键可以使我们的执行逻辑进入函数内部
Ctrl+F5开始执行不调试,如果你想让程序直接运行起来而不调试就可以直接使用

3.3 调试的时候查看程序当前信息

3.3.1 查看临时变量的值

 3.3.2 查看内存信息

3.3.3 查看调用堆栈:反馈函数的调用逻辑

 3.3.4 查看汇编信息

(1)第一种方式:右击鼠标,选择【转到反汇编

 (2)第二种方式

 3.3.5 查看寄存器信息


4.调试实例

实例1:实现代码:求 1!+2!+3! ...+ n! ;不考虑溢出

  1. int main()
  2. {
  3. int i = 0;
  4. int sum = 0;//保存最终结果
  5. int n = 0;
  6. int ret = 1;//保存n的阶乘
  7. scanf_s("%d", &n);
  8. for (i = 1; i <= n; i++)
  9. {
  10. int j = 0;
  11. for (j = 1; j <= i; j++)
  12. {
  13. ret *= j;
  14. }
  15. sum += ret;
  16. }
  17. printf("%d\n", sum);
  18. return 0;
  19. }

输入:5

输出:34863

可以发现,此时代码存在运行错误

1. 首先推测问题出现的原因。初步确定问题可能的原因最好
2. 实际上手调试很有必要
3. 调试的时候我们心里有数

  1. int main()
  2. {
  3. int i = 0;
  4. int sum = 0;//保存最终结果
  5. int n = 0;
  6. int ret = 1;//保存n的阶乘
  7. scanf_s("%d", &n);
  8. for (i = 1; i <= n; i++)
  9. {
  10. int j = 0;
  11. for (j = 1; j <= i; j++)
  12. {
  13. ret *= j;
  14. }
  15. sum += ret;
  16. //修改
  17. ret = 1;
  18. }
  19. printf("%d\n", sum);
  20. return 0;
  21. }

输入:5

输出:153

实例2:

  1. int main()
  2. {
  3. int i = 0;
  4. int arr[10] = { 0 };
  5. for (i = 0; i <= 12; i++)
  6. {
  7. arr[i] = 0;
  8. printf("hehe\n");
  9. }
  10. return 0;
  11. }

输出:无限个hehe循环

这是因为数组越界导致在修改arr的值的时候,同时修改掉了i的值

 i和arr是局部变量,局部变量是放在内存中的栈区的,栈区内存的使用习惯是先使用高地址空间,再使用低地址空间

而数组随着下标的增长,地址是由低变高变化的

环境两个变量之间间隔的整形数
VC6.0 0
gcc1
vs13-192

实例3:在Linux x86-64 gcc环境下,下面的程序会出现什么问题?运行结果是什么?为什么?

  1. int main(int argc,char* argv[])
  2. {
  3. long i;
  4. long a[16];
  5. for (int i = 0; i <= 17; i++)
  6. {
  7. a[i] = 0;
  8. printf("%d", i);
  9. }
  10. return 0;
  11. }

结果:死循环

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/365846
推荐阅读
相关标签
  

闽ICP备14008679号