目录
这是Grace Hopper在1947年在Markll机器运行程序时发现的第一个程序"bug",被贴在她的本子上了。它真的是个虫子。这只夹扁的小飞蛾在MarkⅡ计算机的继电器触点里,它"卡"住了机器的运行,导致运行出错,是第一个计算机程序错误。后来人们习惯性的把程序出的问题称为Bug,把排除程序故障叫做Debug(除虫)。
我们再写代码时,不要只会复制,粘贴。对于程序出现的问题,我们要通过调试来解决,不能如下图:
拒绝-迷信式调试! ! ! !
调试(英语:Debugging / Debug),又称除错,是发现和减少计算机程序或电子仪器设备中程序错误的一个过程。
调试的基本步骤·发现程序错误的存在:
Debug 通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序。
Release称为发布版本,它往往是进行了各种优化,使得程序在代码大小和运行速度上都是最优的,以便用户很好地使用。
- #include
- int main()
- {
- char *p = "hello xilanhua";
- printf("%s\n",p);
- return 0;
- }
上述代码在Debug环境的结果展示
上述代码在Release环境的结果展示
可以发现Release版本下可执行文件比Debug版本下的可执行文件小了很多,说明Release版本做了优化。
看下面这段代码:
- #include
-
- int main()
- {
- int i = 0;
- int arr[10] = { 0 };
- for (i = 0; i <= 12; i++)
- {
- arr[i] = 0;
- printf("hehe\n");
- }
- return 0;
- }
在Debug模式下,x86{32位)环境下,会出现死循环。是因为:
i和数组arr都是局部变量,是在栈区创建的,栈区内存使用特点是先使用高地址空间,而数组又是从低地址往高地址排列,vs编译器变量之间空两个字节,所以当循环执行到arr[12]时会把i的值的值改为0,造成死循环。
如果是debug模式去编译,程序的结果是死循环。如果是release模式去编译,程序没有死循环。那他们之间有什么区别呢?就是因为优化导致的。
注:Linux开发环境调试工具式gdb,后期课程会讲
在环境中选择Debug选项,才能使代码正常调试
最常使用的几个快捷键︰
F5
启动调试,经常用来直接调到下一个断点处。没有断点会让程序运行完。
F9
创建断点和取消断点断点的重要作用,可以在程序的任意位置设置断点。这样就可以使得程序在想要的位置随意停止执行,继而一步步执行下去。
F10
逐过程,通常用来处理一个过程,一个过程可以是一次函数调用,或者是一条语句。
F11
逐语句,就是每次都执行一条语句,但是这个快捷键可以使我们的执行逻辑进入函数内部(这是最长用的)。
CTRL+ F5
开始执行不调试,如果你想让程序直接运行起来而不调试就可以直接使用。
查看临时变量的值:
调试开始后,用于观察变量的值
在调试开始之后开始观察内存信息
多多动手,尝试调试,才能有进步
—定要熟练掌握调试技巧。
初学者可能80%的时间在写代码,20%的时间在调试。但是一个程序员可能20%的时间在写程序,但是80%的时间在调试。
我们所讲的都是一些简单的调试。以后可能会出现很复杂调试场景∶多线程程序的调试等。·多多使用快捷键,提升效率。
优秀的代码∶
常见的coding技巧∶
const的用法
const修饰指针的时候
当const 放在*的左边,限制的指针指向的内容,不能通过指针变量改变指针指向的内容,但是指针变量的本身是可以改变的
当const 放在*的右边,限制的指针变量本身,指针变量的本身是不能改变的,但是指针指向的内容是可以通过指针来改变的
- #include
- int main()
- {
- int m = 10;
- //const 可以修饰指针
- int n = 100;
- const int* p = &m;//*p的值不可改变
- //int* const p = &m; //p的指向不能改变
- //*p = 0;
- //p = &n;
- return 0;
- }
模拟实现库函数: strcpy。示例:
- #include
- #include
- #include
//assert 断言 -
- void my_strcpy(char* dest, const char* src)//const 使为了防止*src被修改
- {
- //断言
- assert(dest != NULL);//条件为假报错,会直接提示是这里出错
- assert(src != NULL);
-
- char* ret = dest;
-
- while (*src != '\0')
- {
- *dest = *src;
- dest++;
- src++;
- }
- *dest = *src;//拷贝 \0
-
- //while (*dest++ = *src++);//等同于上面代码
-
- return ret;
- }
-
-
- int main()
- {
- char arr1[] = "hello world";
- char arr2[20] = "*************";
-
- //会把原字符串的'\0'拷贝过来
- my_strcpy(arr2, arr1);//返回的是目的地的地址
- printf("%s\n", arr2);
- printf("%s\n", strcpy(arr2, arr1));//链式访问
- return 0;
- }
常见的错误分类:
编译型错误
直接看错误提示信息,双击可以定位,解决问题。或者凭借经验就可以搞定。相对来说简单。
链接型错误
link 链接期间发生错误。看错误提示信息,主要在代码中找到错误信息中的标识符,然后定位问题所在。一般是标识符名不存在或者拼写错误。
运行时错误
借助调试,逐步定位问题。最难搞。
最后,今天是第一次恋爱和平分手的日子,为期78天,祝自己以后每天都快乐。
本篇结束