适合初学者的超详细实用调试技巧(下)
我们日常写代码的时候,常常会遇到bug的情况,这个时候像我这样的初学者就会像无头苍蝇一样这里改改那里删删,调试的重要性也就显现出来,这篇文章接着上文来讲解。
上文地址:(8条消息) 适合初学者的超详细实用调试技巧(上)_陈大大陈的博客-CSDN博客
大概分为以下几个部分:
5. 一些调试的实例。
6. 如何写出好(易于调试)的代码。
7. 编程常见的错误。
话不多说,现在开始!
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;
}
我们输入1和2时,结果并没有错误。
这时候我们如果输入3,期待输出9,但实际输出的是15。
为什么呢?
- 首先推测问题出现的原因。初步确定问题可能的原因最好。
- 实际上手调试很有必要。
- 调试的时候我们要做到心里有数
我们小试牛刀调试一下,首先分析问题所在。 编译器没有报错,说明代码没有语法的问题。

输入3,按f11逐语句进行调试。

一次循环下来,sum和ret都变为1,i变为1。

第二次循环下来,仍然看不到什么问题,阶乘和其和也都正确。

在第三次循环,我们发现ret增长的速度十分的快,这才发现ret的值并没有重置为1 。
我们通过调试发现了错误,写出了正确的代码:
#include<stdio.h>
int main()
{int i = 0;int sum = 0;int n = 0;int ret = 1;scanf("%d", &n);for (i = 1; i <= n; i++){int j = 0;ret = 1;//将ret=1置于循环里for (j = 1; j <= i; j++){ret *= j;}sum += ret;}printf("%d\n", sum);return 0;
}
5.2.实例2
给出一个数组越界访问的例子,出自《C陷阱与缺陷》 。
#include <stdio.h>
int main()
{int i = 0;int arr[10] = {0};for(i=0; i<=12; i++){arr[i] = 0;printf("hehe\n");}return 0;
}

如图,程序会死循环打印hehe。
上一个代码可以不用调试看出来错误,但是这个是无法看出来的,只能调试来看。

可以看到到这一步为止都十分正常,也就是i==12之前是正常的。
然而这一步之后,i和arr[i]就一同变成了0。

知道了问题所在,我们这次通过地址来调试看看。

可以看到,当i==12时,i的地址和arr[i]的地址是相同的,也就是说,它们在栈区所开辟的空间相同。这样导致的结果就是, arr[i] = 0的操作将i也一同变成了0,导致死循环。
6. 如何写出好(易于调试)的代码。
6.1 优秀的代码
优秀的代码应该满足以下条件。
1. 代码运行正常
2. bug很少
3. 效率高
4. 可读性高
5. 可维护性高
6. 注释清晰
7. 文档齐全
为了达到这样的条件,我们可以使用以下常见的coding技巧:
1. 使用assert
2. 尽量使用const
3. 养成良好的编码风格
4. 添加必要的注释
5. 避免编码的陷阱。
6.2 示范
我们来模拟实现库函数strcpy:
函数的参数形式char* strcpy(char*destination,const char*source);
该参数说明了strcpy返回类型是char类型的指针,将源头(不能被改)拷贝到目的地。
strcpy特点和strlen类似,遇到‘\0’就停止。
比较容易想到的写法是:
#include<stdio.h>
#include<string.h>
#include<assert.h>
void my_strcpy(char* a, char* b)
{while (*a != '\0'){*b = *a;b++;a++;}*b = *a;
}
int main()
{char a[] = "abcdef";char b[10];my_strcpy(a, b);printf("%s", b);return 0;
}
虽然可以实现strcpy函数的内容,但是优化不怎么样,将简单的功能写的非常麻烦,且无法避免空指针的情况,下面为优化后的写法:
#include<stdio.h>
#include<assert.h>
void my_strcpy(const char a[],char b[])
{assert(a!=NULL&&b!=NULL);//断言函数来避免空指针的情况while (*b++ = *a++){;//当*a为\0的时候,while里的值为假,跳出循环}
}
int main()
{char a[] = "abcdef";char b[10]="";//定义第二个数组来拷贝数组my_strcpy(a,b);printf("%s", b);return 0;
}
值得注意的是,assert断言函数和const的使用,可以大大增加代码的安全性。
6.3 const的作用
#include <stdio.h>
//代码1
void test1()
{int n = 10;int m = 20;int *p = &n;*p = 20;//ok?p = &m; //ok?
}
void test2()
{//代码2int n = 10;int m = 20;const int* p = &n;*p = 20;//ok?p = &m; //ok?
}
void test3()
{int n = 10;int m = 20;int *const p = &n;*p = 20; //ok?p = &m; //ok?
}
int main()
{//测试无cosnt的test1();//测试const放在*的左边test2();//测试const放在*的右边test3();return 0;
}
结论:
const修饰指针变量的时候:
1. const如果放在*的左边,修饰的是指针指向的内容,保证指针指向的内容不能通过指针来改 变。但是指针变量本身的内容可变。
2. const如果放在*的右边,修饰的是指针变量本身,保证了指针变量的内容不能修改,但是指 针指向的内容,可以通过指针改变。
6.4. const的实例
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int my_strlen(const char a[])//用const来使代码更加安全
{int count = 0;while (* a++ != '\0'){count++;}return count;
}
int main()
{char a[]="abcdef";int b = my_strlen(a);printf("%d", b);return 0;
}
7.编程常见的错误
7.1 编译型错误
直接看错误提示信息(双击),解决问题。或者凭借经验就可以搞定。相对来说简单。
如:逗号的使用,分号的添加,括号的对应,各类操作符的使用,库函数的使用格式等。
对于这种问题,我们可以直接通过错误列表的提示来定位问题所在,解决问题。

7.2 链接型错误
看错误提示信息,主要在代码中找到错误信息中的标识符,然后定位问题所在。一般是标识符名不存在或者拼写错误。
如:变量、头文件的包含,文件的引入,常量和宏的定义,库函数名的拼写,自定义函数名的一致等等。
就例如将main写错为mian这样的错误。

7.3 运行时错误
借助调试,逐步定位问题。最难搞。
如:栈溢出,逻辑漏洞,未指针的越界访,未初始化的变量,字符串溢出,数组越界,重复释放内存,使用无效的指针等等。
就像上文里的几个例子。
对于这样的问题,可以通过调试来解决。
说了这么多,调试的章节终于结束了!
希望大家都能成为20%的时间在写程序,但是80%的时间在调试的程序员!

相关文章:
适合初学者的超详细实用调试技巧(下)
我们日常写代码的时候,常常会遇到bug的情况,这个时候像我这样的初学者就会像无头苍蝇一样这里改改那里删删,调试的重要性也就显现出来,这篇文章接着上文来讲解。 上文地址:(8条消息) 适合初学者的超详细实用调试技巧&…...
C# String与StringBuilder 的区分
重点 1)它是比较的栈里面的值是否相等(值比较) 2)Equals它比较的是堆里面的值是否相等(引用地址值比较) 3)Object.ReferenceEquals(obj1,obj2)它是比较的是内存地址是否相等 问题描述: 今日提交代码时候,被检测工具发出修改建议。遂补充一下知识 1.什么…...
【麒麟】基于GPS北斗卫星技术的NTP网络时间服务器
【麒麟】基于GPS北斗卫星技术的NTP网络时间服务器 【麒麟】基于GPS北斗卫星技术的NTP网络时间服务器 麒麟系统NTP授时方案 设计思路: 在通用的麒麟服务器内部固定一块北斗卫星接收模块并引出卫星天线接口,卫星模块接收北斗卫星数据并解码输出时间数据&…...
“互联网+”下劳动关系认定的现状
1. 劳动关系的认定标准。依据目前我国法律的有关规定, 判定劳动关系存在两种情况:其一, 在有书面劳动合同的情况下, 这时应以书面合同作为认定标准;其二, 在没有书面合同的情况下, 则依据2005年劳社部的《关于确立劳动关系有关事项的通知》来认定, 其中第一条:“用人单位招用劳…...
LPWAN及高效弹性工业物联网核心技术方案
20多年前的一辆拖拉机就是一个纯机械的产品,里面可能并没有电子或者软件的构成;而随后随着软件的发展,拖拉机中嵌入了软件,它能控制发动机的功率及拖拉机防抱死系统;接下来,通过融入各种软件,拖…...
OPTIONS FMTSEARCH
FMTSEARCH 指定要检索的格式目录列表,语法如下:OPTIONS FMTSEARCH(catalog-specification-1<catalog-specification-2 … >);使用PROC FORMAT时可以定义格式目录,LIBRARYlibref或LIBRARYlibref.catalog。格式目录可以是libref或libref.…...
Python3 pip
Python3 pip pip 是 Python 包管理工具,该工具提供了对 Python 包的查找、下载、安装、卸载的功能。 软件包也可以在 https://pypi.org/ 中找到。 目前最新的 Python 版本已经预装了 pip。 注意:Python 2.7.9 或 Python 3.4 以上版本都自带 pip 工具…...
【2023-02-20】JS逆向之翼支付
提示:文章仅供参考,禁止用于非法途径 文章目录前言分析总结前言 真的好久没更了…… 提示:以下是本篇文章正文内容,下面案例可供参考 分析 进到网页,加载两个接口 applyLoginFactor 接口返回一个RSA公钥࿰…...
假如面试官问你Babel的原理该怎么回答
1. 什么是 Babel 简单地说,Babel 能够转译 ECMAScript 2015 的代码,使它在旧的浏览器或者环境中也能够运行。 // es2015 的 const 和 arrow function const add (a, b) > a b;// Babel 转译后 var add function add(a, b) {return a b; };Babel…...
深入Spring底层透析Bean创建过程之拨云见日篇
目录前言一.BeanFactory快速入门1. BeanFactory创建Bean2. BeanFactory和ApplicationContext的关系3. 和ApplicationContext区别(高频问点)4. BeanFactory的继承体系5. ApplicationContext的继承体系二.Bean实例化的基本流程(重点)前言 首先感谢您的阅览࿰…...
8 狗监控的封装
概述 为了保证嵌入式程序能够长时间稳定地运行,需要加入狗监控机制。狗监控的原理为:应用程序需要每隔一段时间来喂狗或保活,如果应用程序崩溃或者内核崩溃,导致长时间无法喂狗,则狗将超时,会自动重启系统。部分IPC芯片提供了硬件狗,对于没有硬件狗的,需要自行实现软件…...
基于卷积神经网络图像风格迁移系统的设计与实现(flask系统)
1.摘要 Leon Gatys 等人研发的深度神经网络使用神经的表达来分离任意图片的内容和风格,为生成艺术图片提供一个神经算法。本文基于Style Transfer算法,使用风格成本函数训练CNN,用卷积神经网络提取图像特征,依次提取内容图像的内…...
【1】linux命令每日分享——mkdir
大家好,这里是sdust-vrlab,Linux是一种免费使用和自由传播的类UNIX操作系统,Linux的基本思想有两点:一切都是文件;每个文件都有确定的用途;linux涉及到IT行业的方方面面,在我们日常的学习中&…...
实例2:树莓派GPIO控制外部LED灯闪烁
实例2:树莓派GPIO控制外部LED灯闪烁 实验目的 通过背景知识学习,了解四足机器人mini pupper搭载的微型控制计算机:树莓派。通过树莓派GPIO操作的学习,熟悉GPIO的读写控制。通过外部LED灯的亮灭控制,熟悉树莓派对外界…...
详解可变形注意力模块(Deformable Attention Module)
Deformable Attention(可变形注意力)首先在2020年10月初商汤研究院的《Deformable DETR: Deformable Transformers for End-to-End Object Detection》论文中提出,在2022CVPR中《Vision Transformer with Deformable Attention》提出应用了De…...
Java数据结构中链表分割及链表排序使用快速排序、归并排序、集合排序、迭代、递归,刷题的重点总结
本篇主要介绍在单链表进行分割,单链表进行分隔并使用快速排序、归并排序、集合排序、迭代、递归等方法的总结,愿各位大佬喜欢~~ 86. 分隔链表 - 力扣(LeetCode) 148. 排序链表 - 力扣(LeetCode) 目录 一…...
音视频基础之音频编码原理简介
一:隐蔽信号 数字音频信号如果不加压缩地直接进行传送,将会占用极大的带宽。例如,一套双声道数字音频若取样频率为44.1KHz,每样值按16bit量化,则其码率为: 244.1kHz16bit1.411Mbit/s 如此大的带宽将给信号…...
【Python--XML文件读写】XML文件读写详解
【Python–XML文件读写】XML文件读写详解 文章目录【Python--XML文件读写】XML文件读写详解1. 前言1.1 介绍1.2 用法2. xml文件内容形式3. xml文件读写3.1 项目框架3.1 写入操作(创建)(create_xml.py)3.2 读取操作(解析…...
GNU make 中文手册 第一二章 概述与介绍
一、第一章:概述 准备知识 在开始我们关于 make 的讨论之前,首先需要明确一些基本概念: 编译:把高级语言书写的代码,转换为机器可识别的机器指令。编译高级语言后生成的指令虽然可被机器识别,但是还不能…...
真的了解HashMap、HashSet吗?做一道测试题试试!
本人博客《HashMap、HashSet底层原理分析》,可以了解hashmap的底层源码实现 测试代码 HashSet底层实际就是一个Hashmap。猜猜下面源码每一个打印结果。 注:user对象重写的hashcode方法,保证name和age一样的情况下hashcode是一样的ÿ…...
前端缓存策略:让你的应用飞起来
前端缓存策略:让你的应用飞起来 一、引言 又到了我这个毒舌工匠上线的时间了!今天咱们来聊聊前端缓存策略这个话题。别以为缓存只是后端的事情,前端缓存同样重要。一个好的缓存策略能够大大提高应用的性能和用户体验,让你的应用飞…...
**发散创新:基于Go语言实现的Raft共识算法实战解析**在分布式系统中,**一
发散创新:基于Go语言实现的Raft共识算法实战解析 在分布式系统中,一致性是核心挑战之一。而Raft共识算法因其简洁性和可理解性,已成为当前主流的分布式一致性协议(如etcd、Consul均采用Raft)。本文将带你深入用Go语言从…...
2025届学术党必备的六大降重复率神器横评
Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 当前,人工智能技术快速发展,这为毕业论文写作提供了新的辅助路径&…...
ncmdumpGUI高效使用指南:NCM文件转换完全掌握
ncmdumpGUI高效使用指南:NCM文件转换完全掌握 【免费下载链接】ncmdumpGUI C#版本网易云音乐ncm文件格式转换,Windows图形界面版本 项目地址: https://gitcode.com/gh_mirrors/nc/ncmdumpGUI 一、建立NCM转换认知体系 1.1 理解NCM文件加密机制 …...
终极宝可梦随机化指南:Universal Pokemon Randomizer ZX 完全使用教程
终极宝可梦随机化指南:Universal Pokemon Randomizer ZX 完全使用教程 【免费下载链接】universal-pokemon-randomizer-zx Public repository of source code for the Universal Pokemon Randomizer ZX 项目地址: https://gitcode.com/gh_mirrors/un/universal-po…...
人生苦难的本质的庖丁解牛
“人生苦难的本质”,常被误解为“命运的不公”、“物质的匮乏”或“肉体的疼痛”。 但本质上,苦难并非来自外部世界的客观事件,而是源于**“内在预期”与“外在实相”之间的剧烈摩擦**,是**“有限的自我”试图掌控“无限的无常”时…...
TVA:未来无人车间和智能工厂的质检中枢
「本文已用流量券推广,欢迎收藏 关注」当制造业加速迈向智能化,现代企业的竞争已从产能规模转向技术实力与品质管控能力。AI智能体视觉检测系统(TVA)作为智能制造的核心技术之一,正在成为企业构建智能工厂的关键支撑&…...
技术主管如何降低TVA系统全生命周期成本
对于企业技术主管而言,TVA系统在汽车零部件焊接点检测中的应用,不仅要保障检测质量与效率,还要做好成本管控,避免出现“前期采购成本过高”“后期运维成本失控”“资源浪费”等问题。当前,不少企业因缺乏科学的成本管控…...
OpCore Simplify终极指南:30分钟完成黑苹果智能配置的完整解决方案
OpCore Simplify终极指南:30分钟完成黑苹果智能配置的完整解决方案 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 对于想要体验macOS系统…...
解锁3大核心能力:Path of Building完全掌握指南
解锁3大核心能力:Path of Building完全掌握指南 【免费下载链接】PathOfBuilding Offline build planner for Path of Exile. 项目地址: https://gitcode.com/GitHub_Trending/pa/PathOfBuilding Path of Building(简称PoB)作为《流放…...
