C语言assert断言详解指针(3)

各位少年,大家好,我是博主那一脸阳光,今天分享assert法官的断言,指针宝箱的使用。
前言:如果你在计算机的世界中触犯了语法法规,那么编译器就要上线了,就会出现报错。然而想想我们在现实中设计到经济纠纷什么的,一些自身外部的物资这时候编译器警察就管不了了,没有权限,这时候就需要一些内部法官来管理了,今天我们介绍一个C语言中的法官assert,它是C语言中的库函数。

assert的法官的介绍和宣判

assert叫做“断言”什么是断言“呢?请看下面代码示例。

int a=10;
int b=20;
assert(a==b)//如果不为真,就报错

在这里插入图片描述
大家看到了吧,assert非常的霸道,通常assert在指针的时候常用到,接下来介绍一下函数原型和函数参数。

#include <assert.h>
void assert( int expression );

如果你想要使用这个函数的话,你就要使用它对应的头文件assert.h,返回返回值是void,函数的参数是整形的表达式,如果表达式结果为执行),如果为程序终止)。
assert通常用来测试代码测试运行和指针的运算往往比if语句更加简洁方便。
如果最后我们测试成功以后,该怎么屏蔽assert呢?很显然这里就需要一个了,我们在头文件处加上#define NDEBUG这样就会禁用所有assert语句了。一般我们在Debug版本上的使用,但是Release版本中选择禁用assert版本中已经优化掉了(在VS是这样的),这样在Debug版本中大大节省的时间。

指针的使用和传址调用

函数调用分为两种(传址调用)(——————————(传值调用
传值调用,大家可以理解为传递的值。

#include<stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int a = 10;int b = 20;int ret = Add(a, b);printf("%d\n", ret);return 0;
}

当这段代码调用add函数的时候,add函数调用,传值调用,传入函数的时候,传入的是a和b的变量。

传址调用是什么呢?

传址调用:比方说有两个值,传址就是函数在调用的时候,传递的是这两个值的地址。传值调用是就是传入的两个值,就像一个展览的复品一样,一个副本。

通过传址调用模拟实现strlen函数
#include<stdio.h>
size_t my_strlen(const char *s)
{int count = 0;while (*s != '\0'){count++;s++;}return count;
}
int main()
{char arr[] = "abcdef";size_t len = my_strlen(arr);printf("%zd\n", len);return 0;
}

size_t原型是unsigned int无符号整形,意思是说不能有负号,只能有正数。大家这里理解就可以了。
传址调用和传值调用。
那很显然大家就疑惑了?之前我博客有一篇不需要传址调用模拟实现strlen函数的例子,
那么很显然传值调用传址调用有什么区别,有什么用途?

传址调用实际案例

大家应该还记得我们之前举例子交换两个数字的例子吧?但是我们想要在自定义函数交换?我们也是一样的思路的
但是我们需要用到传址,很显然传值调用不可能实现我们的交换。


#include<stdio.h>
void Swap(int x, int y)
{int z = 0;z = x;x = y;y = z;
}
int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);printf("交换前:a=%d b=%d\n", a, b);Swap(a,b);printf("交换后:a=%d b=%d\n", a, b);return 0;
}
传参调用函数时

函数的实参传给形参时,形参是实参的一份临时拷贝 形参有自己独立的空间 对形参有自己独立的空间 对形参的修改不会影响实参! ```

#include<stdio.h>void Swap2(int* pa, int* pb)
{int z = 0;z = *pa;*pa = *pb;*pb = z;
}
int main()
{int a = 0;int b = 0;scanf("%d %d", &a, &b);printf("交换前:a=%d b=%d\n", a, b);Swap2(&a, &b);printf("交换后:a=%d b=%d\n", a, b);return 0;
}

我们可以看到实现成Swap2的⽅式,顺利完成了任务,这⾥调⽤Swap2函数的时候是将变量的地址传
递给了函数,这种函数调⽤⽅式叫:传址调⽤。
传址调⽤,可以让函数和主调函数之间建⽴真正的联系,在函数内部可以修改主调函数中的变量;所
以未来函数中只是需要主调函数中的变量值来实现计算,就可以采⽤传值调⽤。如果函数内部要修改
主调函数中的变量的值,就需要传址调⽤。

深入理解指针数组

数组名的理解

首先我们要明白一点数组名是数组首元素的地址,这句话是什么意思呢?
比方说

int arr[10]={0,1,2,3,4,5,6,7,8,9,10];
printf("%p",arr);
printf("%p",&arr[0]);

在这里插入图片描述大家看到了,数组名是第一个元素的地址就是0的地址,我们通过上面代码,解释了数组名是首元素的地址,也就是第一个元素。

当然有例外

sizeof(数组名),sizeof中单独放数组名,这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节。

int arr[10]={0,1,2,3,4,5,6,7,8,9,10);
printf("%d\n",sizeof(arr));
printf("%d\n",sizeof(arr[0]));

&数组名,这里的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组首元素的地址是有区别的)

printf("%d",&arr);

除此之外,任何地方使用数组名,数组名表示整个数组的地址。看下面代码例子。

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };printf("%d\n",sizeof(arr));printf("&arr[0] = %p\n", &arr[0]);printf("&arr[0] = %p\n", &arr[0]+1);printf("arr = %p\n", arr);printf("arr = %p\n", arr+1);printf("&arr = %p\n", &arr);printf("&arr = %p\n", &arr+1);return 0;
}
 &arr[0] = 0077F820
&arr[0]+1 = 0077F824
arr = 0077F820
arr+1 = 0077F824
&arr = 0077F820
&arr+1 = 0077F848

这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是
⾸元素的地址,+1就是跳过⼀个元素。 但是&arr 和 &arr+1相差40个字节,这就是因为&arr是数组的地址,+1
操作是跳过整个数组的。 到这⾥⼤家应该搞清楚数组名的意义了吧。 数组名是数组⾸元素的地址

使用指针访问数组

#include<stdio.h>
int main()
{
int arr[10]={ 0 } ;
int i=0;
int sz=sizeof(arr)/sizeof(arr[0]);
int *p=arr;
for(i=0;i<sz;i++)
{
scanf("%d",p+i);
}
for(i=0;i<sz;i++)
{
printf("%d ",*(p+i));
}
return 0;

上面我们通过指针完成对数组输入输出,p每次加i,都访问数组下一个元素的地址。接下来介绍打印数组指针的不同写法i[arr]-->*(arr+i)-->arr[i]

一维数组的传参的本质

#include<stdio.h>
void test(int arr[])//大小可以不写,因为数组传参是传递的是首元素也就是第一个元素。
{
int sz=sizeof(arr)/sizeof(arr[0]);
printf("%d\n",sz);
}
int main()
{
int arr[10]={0};
test(arr);
return 0;
}
数组传参的时候,传递的是并然是数组
传递的是数组首元素的地址
还记得我们之前说过数组的地址是连续的
123。这样的+1就能找到下一个地址

接下来介绍一个算法叫做冒泡排序

冒泡排序

写一个函数,对一个整形数组的数据进行排序
排序方法很多:
1.冒泡排序
2.选择排序
3.插入排序
4.希尔排序
5.快速排序

冒泡排序(Bubble Sort)
思想:相邻的两个元素比较,如果不满足顺序就交换!![在这里插入图片描述

在这里插入图片描述

void BubbleSort(int *arr, int sz)
{int i = 0;for (i = 0; i < sz - 1; i++){int j = 0;for (j = 0; j < sz-i-1; j++){if (arr[j] > arr[j + 1]){int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}
}
int main()
{int arr[] = {9,8,7,6,5,4,3,2,1,0 };int sz = sizeof(arr) / sizeof(arr[0]); BubbleSort(arr, sz);return 0;
}

上面第一个for循环每一次,就代表排序成功一个数字,第二个for循环个数减去i(i已经排序数字),哪个就是1本身的数字比如说这次要排序九,减一就是减去它本身。
好了,这次我们分享到这里 下篇我会分享如何更加深入理解指针。

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

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

相关文章

仰暮计划|“一周一顿的玉米面和白面蒸的糕点,是当时所能吃到的极好的食物”

平淡又记忆深刻的一生 口述人&#xff1a;元奶奶 整理人&#xff1a;宋佳音 口述人基本信息&#xff1a;女 出生于1958年&#xff0c;今年65周岁&#xff0c;祖籍东北&#xff0c;现定居于上海&#xff0c;已从制药厂退休十余年。 元奶奶的自述&#xff1a; 我出生于1958年…

B样条基函数

​定义&#xff1a;令U{u0,u1,…,um}是一个单调不减的实数序列&#xff0c;即ui≤ui1&#xff0c;i0&#xff0c;1&#xff0c;…&#xff0c;m-1。其中&#xff0c;ui称为节点&#xff0c;U称为节点矢量&#xff0c;用Ni,p(u)表示第i个p次&#xff08;p1阶&#xff09;B样条基…

asp.net core通过读取配置文件来动态生成接口

如果希望接口是每次通过配置文件生成的&#xff0c;这样设计一些低代码的方式来获得接口。 系统目录结构&#xff1a; 启动配置代码&#xff1a; using Microsoft.AspNetCore.Hosting; using System.Configuration; using System.Data.Entity; using Swashbuckle.AspNetCore.…

[Python] 什么是PCA降维技术以及scikit-learn中PCA类使用案例(图文教程,含详细代码)

什么是维度&#xff1f; 对于Numpy中数组来说&#xff0c;维度就是功能shape返回的结果&#xff0c;shape中返回了几个数字&#xff0c;就是几维。索引以外的数据&#xff0c;不分行列的叫一维&#xff08;此时shape返回唯一的维度上的数据个数&#xff09;&#xff0c;有行列…

【学网攻】 第(14)节 -- 动态路由(EIGRP)

系列文章目录 目录 系列文章目录 文章目录 前言 一、动态路由EIGRP是什么&#xff1f; 二、实验 1.引入 实验步骤 实验拓扑图 实验配置 看到D开头是便是我们的EIGRP动态路由 总结 文章目录 【学网攻】 第(1)节 -- 认识网络【学网攻】 第(2)节 -- 交换机认识及使用【学…

阿里云智能集团副总裁安筱鹏:企业数字化的终局是什么?

以下文章来源于数字化企业 &#xff0c;作者安筱鹏博士 回答数字化终局追问的起点是&#xff0c;企业需要重新定义我是谁。成为有竞争力的行业领导厂商&#xff0c;你应当成为一个客户运营商&#xff0c;即能够实时洞察、实时满足客户需求&#xff0c;追求极致的客户体验。而要…

版本管理工具git: 谨慎使用git中的撤回操作

文章目录 一、背景二、解决方案1、步骤一2、步骤二 三、参考 一、背景 昨天代码分支提交错了&#xff0c;idea中使用了如下操作&#xff0c;结果代码不见了 二、解决方案 1、步骤一 使用git reflog命令&#xff0c;查看提交记录&#xff0c;找到之前commit操作的哈希值 …

雾锁王国Enshrouded个人专服怎么搭建?怎么多人联机一起玩?

前面跟大家分享了『零基础也可以10秒开服幻兽帕鲁&#xff0c;只需要鼠标点击几下即可自动部署』一文&#xff0c;大家可以快速搭建属于自己的幻兽帕鲁专服。现在我们同样可以在10秒搭建雾锁王国专服&#xff0c;让您与朋友一起合作战胜强大的Boss。 雾锁王国&#xff1a;你是火…

Docker本地部署可编辑开源导航页并发布公网分享好友可访问

文章目录 1. 使用Docker搜索镜像2. 下载镜像3. 查看镜像4. 启动容器5. 浏览器访问6. 远程访问6.1 内网穿透工具安装6.2 创建远程连接公网地址6.3 使用固定二级子域名地址远程访问 今天和大家分享如何使用Docker本地部署一个开源的简约风格网址导航页&#xff0c;支持五种搜索引…

基于C++的面向对象程序设计:类与对象的深入剖析

面向对象程序设计的基本特点 面向对象程序设计的基本特点包括&#xff1a;抽象、封装、继承、多态。 抽象 抽象是指对具体问题或对象进行概括&#xff0c;抽出其公共性质并加以描述的过程。一般情况抽象分为数据抽象和行为抽象&#xff0c;其中数据抽象是指一个对象区别于另…

C++/数据结构:二叉搜索树的实现与应用

目录 一、二叉搜索树简介 二、二叉搜索树的结构与实现 2.1二叉树的查找与插入 2.2二叉树的删除 2.3二叉搜索树的实现 2.3.1非递归实现 2.3.2递归实现 三、二叉搜索树的k模型和kv模型 一、二叉搜索树简介 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0…

IDEA的properties默认编码是UTF-8但是不显示中文

问题描述 今天打开IDEA项目&#xff0c;发现messages_zh_CN.properties不显示中文了 但奇怪的是target下的文件就是展示的中文 而且我IDEA已经配置了编码格式是UTF-8了 使用nodepad打开源文件&#xff0c;也是展示编码格式是UTF-8 &#xff08;打开target下的文件&#xff0c;…

硅谷仿制网易云小程序登录问题解决

引言&#xff1a; 尚硅谷视频中的登录拥挤问题&#xff0c;导致无法登录&#xff0c;所以&#xff0c;我去gitee上从新找了个别人搞好点网易云的api接口【也是比较出名的那个&#xff0c;不想用不行啊&#xff0c;我也试过很多方法都不行】 接口详解网址&#xff1a;网易云音乐…

python基础——池

池的介绍&#xff1a; 提前创建进程池&#xff0c;防止创建的进程数量过多导致系统性能受到影响&#xff0c;在系统执行任务时&#xff0c;系统会使用池中已经创建进程/线程&#xff0c;从而防止资源的浪费&#xff0c;创建的进程/线程可以让多个进程使用&#xff0c;从而降低…

TypeScript Array(九) 数组

1.TypeScript Array 数组 1.1. 描述 数组对象是使用单独的变量名来存储一系列的值。  &emsp比如&#xff0c;你现在有一组数据&#xff0c;存单独变量如下&#xff1a; var data1"Android"; var data2"Java"; var data3"Harmony";那如果有…

南昌市青山湖、滕王阁、洛阳路隧道FM调频广播集群通信调度系统应用案例

一、用户需求 青山湖隧道&#xff0c;是南昌市一条东西走向的城市主干道&#xff0c;隧道为双向6车道&#xff0c;长1070米&#xff0c;其中湖底暗埋段为550米&#xff0c;净高5.45米&#xff0c;两孔每孔净宽12.4米。 滕王阁隧道是南昌市沿江北大道与沿江中大道连通工程&#…

计算机网络实验二

目录 实验二 交换机的基本配置 1、实验目的 2、实验设备 &#xff08;1&#xff09;实验内容&#xff1a; &#xff08;2&#xff09;练习&#xff1a; 1.实验内容一&#xff1a;&#xff08;交换机的配置方式&#xff09; 2.实验内容二&#xff1a;&#xff08;交换机…

云打印怎么收费?云打印需要付费吗?

随着云打印概念的火热发展&#xff0c;很多有打印需求的App或者个人用户都想使用易绘创云打印服务。那么易绘创云打印怎么收费&#xff1f;云打印需要付费吗&#xff1f;今天就带大家来了解一下。 云打印怎么收费&#xff1f;云打印需要付费吗&#xff1f; 很多有打印需求的小…

正则表达式 与文本三剑客(sed grep awk)

一&#xff0c;正则表达式 &#xff08;一&#xff09;正则表达式相关定义 1&#xff0c;正则表达式含义 REGEXP&#xff1a; Regular Expressions&#xff0c;由一类特殊字符及文本字符所编写的模式&#xff0c;其中有些字符&#xff08;元字符&#xff09;不表示字符字面意…

EMQX 性能调优:TCP SYN 队列与 Accept 队列

在上一篇博客&#xff08;EMQX 性能调优&#xff1a;最大连接与文件描述符&#xff09;&#xff0c;我们深入研究了 MQTT 连接与文件描述符之间的关系&#xff0c;介绍了如何修改文件描述符相关的内核参数来突破默认的最大连接数量限制。 但你可能会发现&#xff0c;在某些情况…