精通 VS 调试技巧,学习与工作效率翻倍!
欢迎大家来到贝蒂大讲堂
养成好习惯,先赞后看哦~
所属专栏:C语言学习
贝蒂的主页:Betty‘s blog
1. 什么是调试
当我们写代码时候常常会遇见输出结果不符合我们预期的情况,那这时候我们该怎么办呢?
这时候我们就需要调试(debug),调试简单来说就是去寻找问题,找到错误原因,修改代码的过程。
2. Debug和Release

- 在VS编译器中有着这Debug和Release两个选项,他们分别是调试版本与发布版本,那这两者有什么区别么,我们可以看看下表
| 名称 | 优点 | 缺点 |
|---|---|---|
| Debug | 包含调试信息,易于调试 | 未做任何优化,体积大 |
| Release | 体积小,编译时对应用程序的速度进行优化 | 无法调试 |
- Debug文件与Release文件(需要代码运行才生成)


通过观察我们可以知晓,Release生成的应用文件要比Debug生成的应用文件小的多
3. 调试快捷键
我们在调试过程中可以使用一些快捷键帮助我们节省时间。
下表列出了比较常用的快捷键以及其功能:
| 快捷键 | 功能 |
|---|---|
| F5 | 启动调试,经常⽤来直接跳到下⼀个断点处,⼀般是和F9配合使⽤。 |
| F9 | 创建断点和取消断点 |
| F10 | 逐过程,通常⽤来处理⼀个过程,⼀个过程可以是⼀次函数调⽤,或者是⼀条语句。 |
| F11 | 逐语句,就是每次都执⾏⼀条语句,但是这个快捷键可以使我们的执⾏逻辑进⼊函数内部。在函数调⽤的地⽅,想进⼊函数观察细节,必须使⽤F11,如果使⽤F10,直接完成函数调⽤。 |
| CTRL+F5 | 开始执⾏不调试,如果你想让程序直接运⾏起来⽽不调试就可以直接使⽤。 |
- 断点的作⽤是可以在程序的任意位置设置断点,打上断点就可以使得程序执⾏到想要的位置暂定执⾏,接下来我们就可以使⽤F10,F11这些快捷键,观察代码的执⾏细节。
4. 监视和内存观察
4.1 监视
在调试的过程中我们,如果要观察代码执⾏过程中上下⽂环境中的变量的值,这时候就要用到监视
比如我们要监视下面这段代码:
int main()
{
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", i);
}
return 0;
}
- 打开监视(提前F11启动调试)


- 监视变量
通过监视变量的变化,能够更好的发现错误信息

4.2 内存
除了探究变量改变状况,我们还可以探究变量在内存中的存储
我们还是以上面那段代码举例:
- 打开内存(提前F11启动调试)


- 内存信息



5. 调试举例
5.1 实例一:简单调试
计算 1!+2!+3!的和
错误代码:
int main()
{
int i = 0;
int ret = 1;
int sum = 0;
for (i = 1; i <= 3; i++)
{
for (int j = 1; j <= i; j++)
{
ret *= j;
}
sum += ret;
}
printf("%d ", sum);
return 0;
}
输出结果: 15
有时候我们不能一下子看出错误,这时候我们就需要调试
第一步分析
1!=1
sum=1
2!=2
sum=3
3!=6
sum=9
监控变量并调试分析
- 第一步没问题,F10继续调试

- 第二步没问题,F10继续调试

- 第三步有问题

- 增加变量继续分析

- 第一步没问题,F10继续调试
为什么ret的值会出现不符合预期的情况呢?我们再次观察代码就会发现ret的值在每次使用后都没有更新,所以出现不符合预期的情况
正确代码:
int main()
{
int i = 0;
int ret = 1;
int sum = 0;
for (i = 1; i <= 3; i++)
{
ret = 1;//更新
for (int j = 1; j <= i; j++)
{
ret *= j;
}
sum += ret;
}
printf("%d ", sum);
return 0;
}
5.2 实例二:深度理解
在VS2022、X86、Debug的环境下,下⾯代码输出结果是什么?
#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("betty\n");
}
return 0;
}
输出结果:死循环打印betty
为什么会出现下面结果呢?这段代码不是越界访问了吗?这时我们又要调试起来
- 数组未越界时,并没有发现问题

- 数组越界时,我们发现当arr[12]=0时,i也变为0,这也是为什么代码会死循环的原因

- 为什么改变arr[12]的值,i也随之改变呢?我们猜测i与arr[12]的地址相同

- 事实也证明我们的猜测是正确的
那为什么会这样呢?我们要知道以下三点:
- 局部变量一般存储在内存中的栈区
- 栈区内存的使⽤习惯是从⾼地址向低地址使⽤的,所以变量i的地址是较⼤的。arr数组的地址整体是⼩于i的地址。
- 数组在内存中的存放是:随着下标的增⻓,地址是由低到⾼变化的。
图像演示如下:

- 原因:在该环境下,arr与i的地址直接刚好差两个整型,当越界访问到arr[12]时刚好与i的地址重合,所以改变arr[12]的值也会改变i的值,代码也就会死循环
注意:
在不同的编译器下可能arr与i空出的空间⼤⼩是不⼀样的,代码中这些变量内存的分配和地址分配是编译器指定的,所以的
不同的编译器之间就有差异了。所以这个题⽬是和环境相关的。栈区的默认的使⽤习惯是先使⽤⾼地址,再使⽤低地址的空间,但是这个具体还是要编译器的实现,⽐如:在VS上切换到X64,这个使⽤的顺序就是相反的,在Release版本的程序中,这个使⽤的顺序也是相反的。
5.3 实例三:断点调试
我们知道了简单调试该如何去调,那断点调试该如何去进行呢?
我们以下面这段代码举例:
int main()
{
int i = 0;
for (i = 0; i < 100; i++)//第一步
{
printf("Betty ");
}
int arr[10] = { 0 };
for (i = 0; i < 10; i++)//第二步
{
//.....
}
return 0;
}
- 如果我们可以确定第一步没有问题,问题出现在第二步,如果一步一步调试的话,要调试100次才会到第二步
- 所以这时候我们就需要用到断点
- 首先我们选中要调试的行数,按下F9打下断点,然后F5开始调试

- 如果觉得第一步i==50时有问题,也可以在第一步设置断点,然后右击断点设置条件

6. 编译器报错
6.1 编译型错误
编译型错误⼀般都是语法错误,这类错误⼀般看错误信息就能找到⼀些蛛丝⻢迹,双击错误信息也能初步的跳转到代码错误的地⽅或者附近。

6.2 链接型错误
链接型错误一般是因为标识符名不存在,拼写错误,头文件未包含,引⽤的库不存在

6.3 运⾏时错误
运行时错误复杂多样,一般需要借助调试的手段才能发现
精通 VS 调试技巧,学习与工作效率翻倍!的更多相关文章
- VS Code:让你工作效率翻倍的23个插件和23个编辑技巧
VS Code:让你工作效率翻倍的23个插件和23个编辑技巧 总结了一些平时常用且好用的 VS Code 的插件和编辑技巧分享出来. 文章详情可查阅我的博客:lishaoy.net ,欢迎大家访问. ...
- Goland 这些技巧,学会开发效率翻倍!
hi, 大家好,我是 hhf. <Goland 这些实操技巧,你可能还不会!>介绍了日常开发中一些比较好用的技巧.本篇文章继续介绍一些其他比较好用的技巧. 自定义结构 tag Goland ...
- 老师傅珍藏多年CAD常用快捷键合集,收藏,工作效率翻倍!
想要熟练操作CAD,做一名出色的CAD绘图员,少不了勤学苦练,还要掌握一些常用的绘图命令以及常用快捷键. 今天就来跟大家分享超全的CAD绘图命令,以及常用快捷键,学会涨工资! 常用快捷键: CTRL快 ...
- python virtual env 使用 jupyter ipython notebook,舒服了, 工作效率翻倍
话不多说,尊重原作者 知乎链接
- 2022必须拥有Chrome扩展程序 - 浏览器插件,让你上网效率翻倍
在Chrome网上应用店中查找扩展程序 2022必须拥有Chrome扩展程序 - 浏览器插件,让你上网效率翻倍 可扩展的Chrome Web浏览器比某些人认识的功能强大得多.您可以自定义浏览体验,使其 ...
- 花1分钟配置远程DEBUG,开发效率翻倍,妹子直呼绝绝子
当把一个工程部署到远程服务器后有可能出现意想不到错误,日志打印过多或者过少都影响问题排查的效率,这个时候可以通过远程调试的方式快速定位bug,提升工作效率.本文主要讲解如何使用Idea开发工具进行远程 ...
- iOS bug调试技巧学习----breakpoint&condition
给断点添加条件 - (void)testCondition2 { NSArray *array = @[@"我们", @"一起", @"来" ...
- 使用EditPlus技巧,提高工作效率(附英文版、自动完成文件、语法文件下载)
http://www.cnblogs.com/JustinYoung/archive/2008/01/14/editplus-skills.html
- 掌握这些springboot的配置方式,让你工作效率翻个倍!
springboot的多种配置方式 java配置主要靠java类和一些注解,比较常用的注解有: @Configuration :声明一个类作为配置类,代替xml文件 @Bean :声明在方法上,将方法 ...
- 会了这十种Python优雅的写法,让你工作效率翻十倍,一人顶十人用!
我们都知道,Python 的设计哲学是「优雅」.「明确」.「简单」.这也许很多人选择 Python 的原因.但是我收到有些伙伴反馈,他写的 Python 并不优雅,甚至很臃肿,那可能是你的姿势不对 ...
随机推荐
- 聊聊ChatGLM6B的微调脚本及与Huggingface的关联
本文首先分析微调脚本trainer.sh的内容,再剖析ChatGLM是如何与Huggingface平台对接,实现transformers库的API直接调用ChatGLM模型,最后定位到了ChatGLM ...
- 关于three.js中的矩阵更新
目录 1. 概述 2. 详解 1. 概述 使用如下代码绘制一个面: 'use strict'; function init() { //console.log("Using Three.js ...
- GaussDB(DWS)发生数据倾斜不要慌,一文教你轻松获取表倾斜率
摘要:GaussDB(DWS)是MPP并行架构,若表的数据存在倾斜情况,会引起一系列性能问题,影响用户体验,严重时可能会引起系统故障.因此能快速获取倾斜的表并整改是GaussDB(DWS)运维管理人员 ...
- 实例讲解FusionInsight MRS RTD 实时决策引擎在医保行业应用
摘要: 通过引入FusionInsight RTD实时决策系统,实现医保费用事前预防.事中控制.事后审核的全流程管理 本文分享自华为云社区<FusionInsight MRS RTD 实时决策引 ...
- MRS HetuEgine的数据虚拟化实践
摘要:华为MRS云原生数据湖平台的HetuEngine就是一款解决大数据时代跨源跨域问题的数据虚拟化引擎. 本文分享自华为云社区<基于华为云原生数据湖MRS HetuEgine的数据虚拟化实践& ...
- 带你认识三种kafka消息发送模式
摘要:在kafka-0.8.2之后,producer不再区分同步(sync)和异步方式(async),所有的请求以异步方式发送,这样提升了客户端效率. 本文分享自华为云社区<kafka消息发送模 ...
- 收钱吧与火山引擎VeDI合作一年后 有了哪些新变化?
更多技术交流.求职机会,欢迎关注字节跳动数据平台微信公众号,回复[1]进入官方交流群 收钱吧正在和火山引擎数智平台(VeDI)跑出一条业务提效新通路. 相关数据显示,收钱吧的日服务人次就近 ...
- 跟着字节AB工具DataTester,5步开启一个实验
更多技术交流.求职机会,欢迎关注字节跳动数据平台微信公众号,回复[1]进入官方交流群 火山引擎A/B测试平台DataTester孵化于字节跳动业务内部,在字节跳动,"万事皆A/B,一切可 ...
- PPT 图片8大操作技巧
如何实现图片的批量导入 插入相册 图片批量导出 7z 直接解压 修改扩展名 -> 解压 PPT 抠图 设备透明色 删除背景色 二次曝光效果 低版本 office 通过,形状来实现 图片融入背景 ...
- SpringBoot 配置 Swagger
Profile Config 多环境不同配置 1.添加 Pom 文件 <dependency> <groupId>io.springfox</groupId> &l ...
