这一篇章接着讲调试,以一个实例来说明:
在这个代码中可以发现运行后是进入了死循环,经过调试调出内存信息后可以得到如下结果:
二者的地址是相同的,也正因如此才会出现死循环。这里就是借助了调试从而发现问题。而出现这种情况的原因,与内存栈区的使用习惯是有关系的。栈区使用是先使用高地址的空间后使用低地址的空间。在这个代码中由于数组的元素只存在10个,继续往后走的时候,恰好就走到了存放i这个变量的地址,因此才会出现这种情况。
但要注意,这个代码再不同的编译器中可能结果也存在差异。但无论差异如何,这个代码都是有问题的,因为它发生了越界的行为。
在日常写代码中要写出优秀且实用的代码,常见的coding技巧是必备的。这里以assert 和const 为例进行介绍,手写一个实现库函数strcpy的代码:
在这个代码中有几点需要说明:
const用于*符号前,修饰的是*b,表示指针所指向的对象不可通过b随意改变。这样修饰的作用在于避免传参时传错位而造成代码bug,但注意b变量中的地址是可以改变的。而const放于*符号后则反之,p不能变 *p可变。
assert即断言。很多时候,在定义一个指针后,可能会出现暂时不赋值的状态。在这种情况下需要将指针定义为空指针(NULL)。但是如果将空指针带入到此函数中进行运行,是会造成问题的。因此,在运行前先先进行断言,若为空指针,程序不会执行且弹出警告,避免造成bug。
再说函数的返回类型。其实在这个代码的函数中,完全可以不返回任何值而只把字符进行调换,但是这里把返回类型修改为char* 类型是为了使代码达成链式访问,大概意思就是更流畅。因为这样有返回值后在主函数内部就可以直接打印,无需再定义一个临时变量。
最后是while的判定条件。while括号内部实际上是先执行后判断,那么就可以如这个代码这样书写,直接在括号内部进行赋值,也就是说当if或者while语句中的条件为赋值语句时,实际上是将赋值后的结果与0进行比较。而在这个代码中,当赋值为斜杠零时,即为零,比较后为假,则不执行。所以赋值完毕。
以上就是这个代码需要注意的地方,那么在搞清楚了后,可以仿造写一个模拟库函数strlen功能的代码:
最后再总结一下编程常见的错误:
1.编译型错误
大多数编译型错误都会在编译器中标红提示,最常见的就是代码相关的语法书写错误。
2.链接型错误
有错误信息提示,主要再代码中找到错误信息中的标识符,然后定位问题所在,一般是标识符名不存在或者拼写错误。
3.运行时错误
这种就是bug,需要调试一步步找出问题。
关于递归与迭代的差异:递归是反复调用函数自身从而实现循环,迭代是调用函数内部一段代码实现循环。
-------------------------------------最后编辑于2023.3.8