p28 vs环境-C语言实用调试技巧

int main()
{
    int i=0;
    for(i=0;i<100;i++)
    {
        printf("%d",i);
    }
}

1.Debug 和Release的介绍

Debug通常称为调试版本,它包含调试信息,并且不做任何优化,便于程序员调试程序。

Release称为发布版本,它往往 是进行了各种优化,使得程序在代码大小和运行速度上都是最优的,以便用户很好的使用。代码:

#include<stdio.h>

int main()

{

        char *p="hello bit";

        printf("%s\n",p);

        return 0;

}

Debug版本的可执行程序   可调试的,因为文件中包含了调试信息

Release版本的可执行程序  

int main(){int i=0;int arr[10]={1,2,3,4,5,6,7,8,9,10};for(i=0;i<=12;i++){printf("hehe\n");arr[i]=0;}system("pause");return 0;
}

//F5-启动调试-和F9配合使用的

int main()
{
    int i=0;
    for(i=0;i<100;i++)
    {
        printf("%d ",i);
    }
        for(i=0;i<100;i++)
    {
        printf("%d ",10-i);
    }
        return 0;

}

2.快捷键 

f5

启动调试,经常用来直接调到下一个断点处。

f9

创建断点和取消断点 断点的重要作用,可以在程序的任意位置设置断点。这样就可以使得程序在想要的位置随意停止执行,继而一步一步执行下去。

f10

逐过程,通常用来处理一个过程,一个过程可以是一次函数调用,或者一条语句

f11

逐语句,就是每次都执行一条语句,但是这个快捷键可以使我们的执行逻辑进入函数的内部(这是最长的)。

ctrl+f5

开始执行不调试,如果你想让程序运行起来而不调试可以直接使用。

3.调试的时候查看程序当前信息

查看临时变量的值

int Add(int x,int y)
{
    return x+y;
}

int main(){

    printf("hehe\n");
    int a=20;
    int b=10;
    int c=Add(a,b);
    return 0;

}

int main()
{
    {
        int tmp=0;
        printf("tmp=%d\n",tmp);
    }
    int arr[10]={0};
    int i=0;
    for(i=0;i<10;i++){
        arr[i]=i;
    }
    return 0;
}

void test2()
{printf("hehe\n");
}
void test1()
{test2();
}
void test()
{test1();
}
int main()
{test();return 0;
}

4.多多动手,尝试调试,才能有进步

  • 一定要熟练掌握调试技巧
  • 初学者可能80%的时间在写代码,20%的时间在调试。但是一个程序员可能20%的时间在写程序,但是80%的时间在调试。
  • 我们所讲的都是一些简单的调试。以后可能会出现更复杂调试场景,多线程程序的调试等 。
  • 多多使用快捷键,提升效率。

一些调试的实例

实例一

实现代码:求1!+2!+3!+...+n!;不考虑溢出。

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;ret=1;for(j=1;j<=i;j++){ret*=j;}sum+=ret;}printf("%d\n",sum);return 0;}

当你输入3的时候,结果得出的是9

1.栈区的默认使用

先使用用高地址处的空间

再使用低地址处的空间

2.数组随着下标的增长

地址是由低到高变化

int main()
{int arr[10]={1,2,3,4,5,6,7,8,9,10};int i=0;//VC6.0环境下<=10就死循环了//gcc编译器<=11就死循环了//vs2013  <=12死循环for(i=0;i<=10;i++){printf("hehe\n");arr[i]=0;}system("pause");return 0;
}

int main()
{int i=0;int arr[10]={1,2,3,4,5,6,7,8,9,10};printf("%p\n",arr);printf("%p\n",&i);//VC6.0环境下<=10就死循环了//gcc编译器<=11就死循环了//vs2013  <=12死循环//system("pause");
//	for(i=0;i<=12;i++)
//	{
//		printf("hehe\n");
//		arr[i]=0;
//	}return 0;
}

如何写出好的代码

优秀的代码:

  1. 代码运行正常
  2. bug很少
  3. 效率很高
  4. 可读性高
  5. 可维护性高
  6. 注释清晰 
  7. 文档齐全

常见的coding技巧

  1. 实用assert
  2. 尽量使用const
  3. 养成良好的编码风格
  4. 添加必要的注释
  5. 避免编码的陷阱

示范:

模拟实现库函数:strcpy


int main()
{//strcpy//字符串拷贝char arr1[]="##############";char arr2[]="bit";strcpy(arr1,arr2);printf("%s\n",arr2);return 0;
}

//6分
void my_strcpy(char *dest,char *src)
{while(*src!='\0'){*dest=*src;src++;dest++;}*dest=*src;//'\0'}int main()
{//strcpy//字符串拷贝char arr1[]="##############";char arr2[]="bit";my_strcpy(arr1,arr2);printf("%s\n",arr1);return 0;
}

满分10分  这个7分

void my_strcpy(char *dest,char *src)

{
    if(dest!=NULL&&src!=NULL)
    {
        while(*dest++=*src++)
        {
                ;
        }

    }
    


}

int main()
{
    //strcpy
    //字符串拷贝
    char arr1[]="##############";
    char arr2[]="bit";
    my_strcpy(arr1,arr2);
    printf("%s\n",arr1);
    return 0;
}

8分
 

#include<assert.h>
void my_strcpy(char *dest,char *src){assert(dest!=NULL);//断言assert(src!=NULL);//断言while(*dest++=*src++){;}}int main()
{//strcpy//字符串拷贝char arr1[]="##############";char arr2[]="bit";my_strcpy(arr1,arr2);printf("%s\n",arr1);return 0;
}

9分当你把左右代码写反的时候,const就起作用了

#include<assert.h>
void my_strcpy(char *dest,const char *src){
assert(dest!=NULL);//断言
assert(src!=NULL);//断言
while(*dest++=*src++){;}}int main()
{
//strcpy
//字符串拷贝
char arr1[]="##############";
char arr2[]="bit";
my_strcpy(arr1,arr2);
printf("%s\n",arr1);
return 0;
}

10分 

#include<assert.h>
char * my_strcpy(char *dest,const char *src){char *ret=dest;
assert(dest!=NULL);//断言
assert(src!=NULL);//断言
//把src指向的字符串拷贝到dest指向的空间,包含'\0'字符
while(*dest++=*src++){;}return ret;}int main()
{
//strcpy
//字符串拷贝
char arr1[]="##############";
char arr2[]="bit";printf("%s\n",my_strcpy(arr1,arr2));
return 0;
}
#include<assert.h>
int my_strlen(const char*str)
{int count=0;assert(str!=NULL);//保证指针的有效性while(*str!='\0'){count++;str++;}return count;
}
int main()
{char arr[]="abcdef";int len=my_strlen(arr);printf("%d\n",len);return 0;
}

#include<assert.h>
int main()
{
    int a=0;
    int *p=&a;
    assert(p!=NULL);
    return 0;
}

链接错误

int Add(int x,int y)
{return x+y;
}
int main()
{int a=10;int b=20;int sum=Add(a,b);printf("%d\n",sum);
}

函数没定义

函数名字写错了

编译型错误

直接靠错误提示信息(双击),解决问题。或者凭借经验可以搞定,相对来说简单。

链接型错误

看错误提示信息,主要在代码中找到错误信息中的标识符,然后定位问题所在 ,一般是表示符名不存在或者拼写错误。

运行时错误

借助调试,逐步定位问题,最难搞。

温馨提示

做一个有心人,积累拍错经验

讲解重点:介绍每种错误怎么产生,出现后,如何解决。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/876722.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PTPD 在 QNX 系统上的授时精度验证与误差排查

文章目录 0. 引言1.关键函数实现2. 验证策略与结果3. 授时误差的排查与解决3. 授时误差的排查与解决4. 结论 0. 引言 PTPD是一种时间同步的开源实现&#xff0c;在不同操作系统上的表现可能存在显著差异。 本文通过在QNX系统上运行PTPD&#xff0c;针对其授时精度进行详细验证…

探索算法系列 - 双指针

目录 移动零&#xff08;原题链接&#xff09; 复写零&#xff08;原题链接&#xff09; 快乐数&#xff08;原题链接&#xff09; 盛最多水的容器&#xff08;原题链接&#xff09; 有效三角形的个数&#xff08;原题链接&#xff09; 查找总价格为目标值的两个商品&…

优化算法:2.粒子群算法(PSO)及Python实现

一、定义 粒子群算法&#xff08;Particle Swarm Optimization&#xff0c;PSO&#xff09;是一种模拟鸟群觅食行为的优化算法。想象一群鸟在寻找食物&#xff0c;每只鸟都在尝试找到食物最多的位置。它们通过互相交流信息&#xff0c;逐渐向食物最多的地方聚集。PSO就是基于这…

【python_将一个列表中的几个字典改成二维列表,并删除不需要的列】

def 将一个列表中的几个字典改成二维列表(original_list,headersToRemove_list):# 初始化一个列表用于存储遇到的键&#xff0c;保持顺序ordered_keys []# 遍历data中的每个字典&#xff0c;添加其键到ordered_keys&#xff0c;如果该键还未被添加for d in original_list:for …

P4009 汽车加油行驶问题题解

P4009 汽车加油行驶问题 紫题&#xff0c;但是DFS。 思路 记忆化搜索&#xff0c;分多钟情况去搜索。 注意该题不用标记&#xff0c;有可能会往回走。 有可能这样走。 代码 #include<bits/stdc.h> #include<cstring> #include<queue> #include<set&g…

redis:清除缓存的最简单命令示例

清除redis缓存命令(执行命令列表见截图) 1.打开cmd窗口&#xff0c;并cd进入redis所在目录 2.登录redis redis-cli 3.查询指定队列当前的记录数 llen 队列名称 4.清除指定队列所有记录 ltrim 队列名称 1 0 5.再次查询&#xff0c;确认队列的记录数是否已清除

配置和连接另一台电脑上的 MySQL 数据库

要配置和连接另一台电脑上的 MySQL 数据库&#xff0c;可以按照以下步骤进行设置&#xff1a; 1. 配置 MySQL 服务器 在目标计算机上&#xff08;192.168.10.103&#xff09;进行以下操作&#xff1a; 修改 MySQL 配置文件&#xff1a; 打开 MySQL 配置文件&#xff08;通常位…

【系统架构设计师】十八、信息系统架构设计理论与实践①

目录 一、信息系统架构概述 二、信息系统架构风格与分类 2.1 信息系统架构风格 2.2 信息系统架构分类 三、信息系统架构模型 3.1 单体应用 3.2 客户机/服务器 3.2.1 二层 C/S 3.2.2 三层 C/S 和 B/S 3.2.3 多层 C/S 和 B/S 3.2.4 MVC 3.3 面向服务架构(SOA)模式 …

Activiti 本地画流程 http://localhost:8080/activiti-app/#/

http://localhost:8080/activiti-app/#/ 1、本地安装了Tomcat 2、本地安装了Activiti 3、拷贝Activiti中这两个文件到Tomcat中的webapps目录下 4、启动startu.bat 5、http://localhost:8080/activiti-app/#/ 账号&#xff1a;admin 密码&#xff1a;test

乐鑫 Matter 技术体验日回顾|全面 Matter 解决方案驱动智能家居新未来

日前&#xff0c;乐鑫信息科技 (688018.SH) 在深圳成功举办了 Matter 方案技术体验日活动&#xff0c;吸引了众多照明电工、窗帘电机、智能门锁、温控等智能家居领域的客户与合作伙伴。活动现场&#xff0c;乐鑫产研团队的小伙伴们与来宾围绕 Matter 产品研发、测试认证、生产工…

Python学习笔记46:游戏篇之外星人入侵(七)

前言 到目前为止&#xff0c;我们已经完成了游戏窗口的创建&#xff0c;飞船的加载&#xff0c;飞船的移动&#xff0c;发射子弹等功能。很高兴的说一声&#xff0c;基础的游戏功能已经完成一半了&#xff0c;再过几天我们就可以尝试驾驶 飞船击毁外星人了。当然&#xff0c;计…

解析西门子PLC的String和WString

西门子PLC有两种字符串类型&#xff0c;String与WString String 用于存放英文数字标点符号等ASCII字符&#xff0c;每个字符占用一个字节 WString宽字符串用于存放中文、英文、数字等Unicode字符&#xff0c;每个字符占用两个字节 之前我搞过一篇解析String的 关于使用TCP-…

Vue3 Pinia的创建与使用代替Vuex 全局数据共享 同步异步

介绍 提供跨组件和页面的共享状态能力&#xff0c;作为Vuex的替代品&#xff0c;专为Vue3设计的状态管理库。 Vuex&#xff1a;在Vuex中&#xff0c;更改状态必须通过Mutation或Action完成&#xff0c;手动触发更新。Pinia&#xff1a;Pinia的状态是响应式的&#xff0c;当状…

Linux内核 mmap内存映射的实现原理

在Linux内核以及Linux系统编程的时候&#xff0c;经常会碰到mmap内存映射&#xff0c;mmap函数是实现高性能编程的一个关键点。本文详细介绍一下mmap实现原理。 虚拟地址映射物理地址 虚拟地址映射物理地址采用的是页表机制&#xff0c;64位CPU采用的是4级页表。 64位CPU虚拟…

鸿蒙 HarmonyOS NEXT端云一体化开发-认证服务篇

一、开通认证服务 地址&#xff1a;AppGallery Connect (huawei.com) 步骤&#xff1a; 1 进入到项目设置页面中&#xff0c;并点击左侧菜单中的认证服务 2 选择需要开通的服务并开通二、端侧项目环境配置 添加依赖 entry目录下的oh-package.json5 // 添加&#xff1a;主要前…

《python程序语言设计》第6章14题 估算派值 类似莱布尼茨函数。但是我看不明白

这个题提供的公式我没看明白&#xff0c;后来在网上找到了莱布尼茨函数 c 0 for i in range(1, 902, 100):a (-1) ** (i 1)b 2 * i - 1c a / bprint(i, round(4 / c, 3))结果 #按题里的信息&#xff0c;但是结果不对&#xff0c;莱布尼茨函数到底怎么算呀。

PyTorch深度学习快速入门(上)

PyTorch深度学习快速入门&#xff08;上&#xff09; 一、前言&#xff08;一&#xff09;PyTorch环境配置&#xff08;二&#xff09;Python编译器的选择&#xff08;三&#xff09;Python学习中的两大法宝函数 二、如何加载数据&#xff08;一&#xff09;Dataset与Dataloade…

轻松学EntityFramework Core--模型创建

一、使用代码优先&#xff08;Code-First&#xff09;创建模型 Code-First 方法是 EF Core 提供的一种用于定义模型的方式&#xff0c;它允许开发人员通过编写 C# 类来定义数据库模式&#xff0c;再通过迁移命令生成数据库表。下面我们来一起看一下代码优先如何使用。 1.1、创…

lua 游戏架构 之 游戏 AI (六)ai_auto_skill

定义一个为ai_auto_skill的类&#xff0c;继承自ai_base类。ai_auto_skill类的目的是在AI自动战斗模式下&#xff0c;根据配置和条件自动选择并使用技能。 lua 游戏架构 之 游戏 AI &#xff08;一&#xff09;ai_base-CSDN博客文章浏览阅读379次。定义了一套接口和属性&#…

【原创】使用keepalived虚拟IP(VIP)实现MySQL的高可用故障转移

1. 背景 A、B服务器均部署有MySQL数据库&#xff0c;且互为主主。此处为A、B服务器部署MySQL数据库实现高可用的部署&#xff0c;当其中一台MySQL宕机后&#xff0c;VIP可自动切换至另一台MySQL提供服务&#xff0c;实现故障的自动迁移&#xff0c;实现高可用的目的。具体流程…