实用调试技巧(1)

简介: 实用调试技巧(1)


1. 什么是bug?

bug原义为虫子,小昆虫;又因为第一次被发现的导致计算机错误的是一只飞蛾,所以bug又译为程序错误。

2. 调试是什么?有多重要?

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

2.1 调试是什么?

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

2.2 调试的基本步骤

  1. 发现程序错误的存在
  2. 以隔离、消除等方式对错误进行定位
  3. 确定错误产生的原因
  4. 提出纠正错误的解决办法
  5. 对程序错误予以改正,重新测试

2.3 Debug和Release的介绍

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

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

#include <stdio.h>
int main()
{
  int arr[10] = { 0 };
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    arr[i] = 10 - i;
  }
  for (i = 0; i < 10; i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}

以上代码在Debug版本中的大小:

以上代码在Release版本中的大小:

此外,只有在Debug环境下才能够进行调试,而在Release环境下是无法进行调试的!!!

3. Windows环境调试介绍

注:linux开发环境调试工具是gdb,后期会进行介绍。

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. 多多动手,尝试调试,才能有进步

  • 一定要熟练掌握调试技巧。
  • 初学者可能80%的时间在写代码,20%的时间在调试;但是一个程序员可能20%的时间在写程序,但是80%的时间在调试。
  • 我们所讲的都是一些简单的调试,以后可能会出现很复杂调试场景:多线程程序的调试等。
  • 多多使用快捷键,提升效率。

5. 一些调试的实例

5.1 实例一

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

#include <stdio.h>
int main()
{
  int i = 0;
  int sum = 0;//保存最终结果
  int n = 0;
  int ret = 1;//保存n的阶乘
  scanf("%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;
}

这时候我们如果输入3,期待输出9,但实际输出的是15。

why?

这里我们就得找我们的问题:

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

通过一步步的调试,我们发现是因为ret没有及时变回1所导致的,正确代码如下:

#include <stdio.h>
int main()
{
  int i = 0;
  int sum = 0;//保存最终结果
  int n = 0;
  int ret = 1;//保存n的阶乘
  
  scanf("%d", &n);//3
  
  for (i = 1; i <= n; i++)//1 2 3
  {
    int j = 0;
    ret = 1;
    
    for (j = 1; j <= i; j++)
    {
      ret *= j;
    }
    
    sum += ret;
  }
  
  printf("%d\n", sum);
  
  return 0;
}

5.2 实例二

研究程序死循环的原因:

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

通过调试,我们发现当i = 12,arr[i] = 0时,arr[12]和i同时变为0,于是我们可以取出arr[12]和i的地址进行观察,发现它们的地址是相同的,因此,当把arr[12]改为0的同时,也把i改成了0,所以造成了死循环。

那么为什么arr[12]和i的地址相同呢?

i和arr是局部变量,是放在内存中栈区上的,栈区内存的使用习惯:先使用高地址处的空间,再使用低地址处的空间;又因为数组随着下标的增长,地址是由低到高变化的。


此外,上文提到过Release版本可以进行各种优化,实例二就能证明:

#include <stdio.h>
int main()
{
  int i = 0;
  int arr[10] = { 0 };
  printf("%p\n", &i);
  printf("%p\n", &arr[0]);
  printf("%p\n", &arr[9]);
  for (i = 0; i <= 12; i++)
  {
    arr[i] = 0;
    printf("hehe\n");
  }
  
  return 0;
}

将以上代码切换到Release版本下运行,就会发现其没有死循环,也没有报错,这就证明了Release版本确实可以进行优化。(Release版本中i的地址比arr[0]的地址要低;变量在内存中开辟的顺序发生了变化,影响到了程序执行的结果)


目录
相关文章
|
Java 调度 C++
ANR分析总结
ANR分析总结
1494 0
ANR分析总结
|
6月前
|
存储 数据安全/隐私保护 开发者
苹果app上架app store 之苹果开发者账户在mac电脑上如何使用钥匙串访问-发行-APP发布证书ios_distribution.cer-优雅草卓伊凡
苹果app上架app store 之苹果开发者账户在mac电脑上如何使用钥匙串访问-发行-APP发布证书ios_distribution.cer-优雅草卓伊凡
224 8
苹果app上架app store 之苹果开发者账户在mac电脑上如何使用钥匙串访问-发行-APP发布证书ios_distribution.cer-优雅草卓伊凡
|
6月前
|
SQL druid Oracle
【YashanDB知识库】yasdb jdbc驱动集成druid连接池,业务(java)日志中有token IDENTIFIER start异常
客户Java日志中出现异常,影响Druid的merge SQL功能(将SQL字面量替换为绑定变量以统计性能),但不影响正常业务流程。原因是Druid在merge SQL时传入null作为dbType,导致无法解析递归查询中的`start`关键字。
|
机器学习/深度学习 人工智能 自然语言处理
【AI 生成式】什么是生成式 AI,它与判别式 AI 有何不同?
【5月更文挑战第4天】【AI 生成式】什么是生成式 AI,它与判别式 AI 有何不同?
【AI 生成式】什么是生成式 AI,它与判别式 AI 有何不同?
|
12月前
|
Python
音乐播放 pygame mp3play 和获取音乐信息的 库from mutagen.mp3 import MP3
音乐播放 pygame mp3play 和获取音乐信息的 库from mutagen.mp3 import MP3
|
存储 SQL 关系型数据库
MySQL数据库进阶第四篇(视图/存储过程/触发器)
MySQL数据库进阶第四篇(视图/存储过程/触发器)
|
存储 关系型数据库 Java
极速体验DolphinScheduler 3.2.1 Standalone 版[一]
极速体验DolphinScheduler 3.2.1 Standalone 版[一]
151 0
|
人工智能 供应链 算法
SWOT分析法:知彼知己的战略规划工具
SWOT分析法是一种用于评估组织、项目、个人或任何其他事物的战略规划工具。SWOT是Strengths(优势)、Weaknesses(劣势)、Opportunities(机会)和Threats(威胁)的缩写。通过分析这四个方面,SWOT分析法可以帮助决策者了解当前情况,并为未来的行动制定策略。
732 0
SWOT分析法:知彼知己的战略规划工具
|
消息中间件 Java 程序员
阿里巴巴高并发架构到底多牛逼?是如何抗住淘宝双11亿级并发量?
众所周知,在Java的知识体系中,并发编程是非常重要的一环,也是面试的必问题,一个好的Java程序员是必须对并发编程这块有所了解的。
|
存储 前端开发 安全
DAPP区块链商城系统开发(方案逻辑)丨区块链DAPP商城系统开发(案例设计)/开发项目/源码部署
 区块链(Blockchain)是一种由多方共同维护,使用密码学保证传输和访问安全,能够实现数据一致存储、难以篡改、防止抵赖的记账技术,也称为分布式账本技术(Distributed Ledger Technology)。从本质上看,区块链是通过去中心化和去信任化,集体维护、分布式存储的可靠数据库。