VS编译器调试技巧和bug处理

目录

1.什么是bug?

2.什么是调试(debug)

3.VS调试快捷键

4.监视和内存观察

4.1监视

4.2 内存

5.动手调试bug

案例一

案例二


1.什么是bug?

bug本意是“昆虫”或“虫子”,现在⼀般是指在电脑系统或程序中,隐藏着的⼀些未被发现的缺陷或问题,简称程序漏洞。

“Bug” 的创始⼈格蕾丝·赫柏(Grace Murray Hopper),她是⼀位为美国海军工作的电脑专家,1947年9⽉9⽇,格蕾丝·赫柏对Harvard Mark II设置好17000个继电器进⾏编程后,技术⼈员正在进行整机运行时,它突然停止了工作。于是他们爬上去找原因,发现这台巨⼤的计算机内部⼀组继电器的触点之间有⼀只⻜蛾,这显然是由于⻜蛾受光和热的吸引,飞到了触点上,然后被⾼电压击死。所以在报告中,赫柏⽤胶条贴上⻜蛾,并把“bug”来表示“⼀个在电脑程序⾥的错误”,“Bug”这个说法⼀直沿⽤到今天。

2.什么是调试(debug)

当我们发现程序中存在的问题的时候,那下⼀步就是找到问题,并修复问题。

这个找问题的过程叫称为调试,英文叫debug(消灭bug)的意思。

调试⼀个程序,首先先是承认出现了问题,然后通过各种手段去定位问题的位置,可能是逐过程的调试,也可能是隔离和屏蔽代码的方式,找到问题所的位置,然后确定错误产⽣的原因,再修复代码,重新测试。

  • Debug 通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序;程序员在写代码的时候,需要经常性的调试代码,就将这里设置为 debug ,这样编译产生的是debug 版本的可执行程序,其中包含调试信息,是可以直接调试的。

  • Release 称为发布版本,它往往是进⾏了各种优化,使得程序在代码大小和进行速度上都是最优的,以便用户很好地使用。当程序员写完代码,测试再对程序进行测试,直到程序的质量符合交付给用户使用的标准,这个时候就会设置为 release ,编译产⽣的就是 release 版本的可执行程序,这个版本是用户使用的,无需包含调试信息等(不能一步步调试结果) 测试人员测试的时候会测试Release版本

对比可以看到从同⼀段代码,编译⽣成的可执行文件的大小,release版本明显要小,而debug版本明显大

3.VS调试快捷键

调试最常使用的几个快捷键:

  • F9:创建断点和取消断点

    断点的作用是可以在程序的任意位置设置断点,打上断点就可以使得程序执行到想要的位置暂停执行,接下来我们就可以使用F10,F11这些快捷键,观察代码的执行细节。

  • F5:启动调试,经常用来直接跳到(运行逻辑处)下⼀个断点处(不需要按部就班的一个个调试),⼀般是 和F9配合使用。用F5跳到该行代码后,用F11一步步的整个过程调试

  • 条件断点:满足这个条件,才触发断点, 我们可以设置一个条件,指定让他跳过之前的循环 当i==5的时候才开始调试,更方便

  • F10:逐过程,通常用来处理⼀个过程,⼀个过程可以是⼀次函数调用,或者是⼀条语句。(如果是一个函数,他会直接执行这个函数的所有语句,而不是一句句调试)

  • F11:逐语句,就是每次都执行⼀条语句,但是这个快捷键可以使我们的执行逻辑进⼊函数内部。在函数调用的地方,想进⼊函数观察细节,必须使用F11,如果使用F10,直接完成函数调用。

  • CTRL + F5:开始执行不调试,如果你想让程序直接运行起来而不调试就可以直接使⽤。

4.监视和内存观察

4.1监视

开始调试后,在菜单栏中【调试】->【窗⼝】->【监视】,打开任意⼀个监视窗口,输⼊想要观察的对象就行。

打开监视窗口:

在窗口中观察值的变化:

4.2 内存

如果监视窗⼝看的不够仔细,也是可以观察变量在内存中的存储情况,还是在【调试】->【窗⼝】->【内存】

打开内存窗口:

在内存窗口中观察数据:

5.动手调试bug

案例一

该代码有问题,我们应该怎么修改呢

//求阶乘 求 1!+2!+3!+4!+...10! 的和,
int main() {
    int n = 0;
    int ret = 1; //返回值
    int sum = 0;//最后的和
    for (n = 1; n <= 10; n++) {
        for (int i = 1; i <= n; i++) {
            ret = ret * i;
        }
        sum += ret;
    }
    printf("%d", sum);
​
    return 0;
}

为什么观察到ret的值为12呢,因为他将上一次算出来的ret值2也乘上去了,所以我们每次都需要将ret的初始值置为1 ret = 1

正确的写法

//求阶乘 求 1!+2!+3!+4!+...10! 的和,
int main() {
    int n = 0;
    int ret = 1; //返回值
    int sum = 0;//最后的和
    for (n = 1; n <= 10; n++) {
        ret = 1;  //应该每次都将ret初始化为1
        for (int i = 1; i <= n; i++) {          
            ret = ret * i;
        }
        sum += ret;
    }
    printf("%d", sum);
    return 0;
}

算法可以进一步优化 只需要循环10次

int main() {
    int n = 0;
    int ret = 1; //返回值
    int sum = 0;//最后的和
    int i = 0;
    for (n = 1; n <= 3; n++) {
​
        ret = ret * n;
        sum += ret;
    }
    printf("%d", sum);
​
    return 0;
}

案例二

在VS2022、X86、Debug 的环境下,编译器不做任何优化的话,下面代码执行的结果是啥?

  • 我们看下图,发现i和arr[12]的地址一模一样,可以知道假如数组越界的时候,i和arr[12]一起变化 永远不会到13 不会跳出循环所以会死循环

  • 最主要的bug就是数组越界,把这个问题解决即可

    • i和arr是局部变量,局部变量是放在内存的栈区,

    • 栈区上的内存的使用习惯是:先使用高地址的空间,再使用低地址的空间

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

      (刚好<=12的时候死循环,11的话不会死循环最后越界退出循环, 低到高一直刚好等于i的地址,然后i和arr地址一样)

    • i和arr中间 空多大空间,完全是取决于编译器实现

    • vc6.0 中 就这个代码 arr和i之间没有空隙

    • gcc 中就这个代码 arr和i之间空一个int

    • vs 中间空的是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变量先创建,arr[]后创建 局部变量在栈区创建,地址是由低到高;当下标不断变大的时候可能刚好覆盖i的地址,导致死循环;

  • 观察数组传参的时候需要注意的问题

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值