深入理解指针(2)(C语言版)

文章目录

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

前言

在上一篇文章中,我们初步了解了指针的基本概念和用法。今天,我们将继续深入探索指针在数组、函数传参以及复杂数据结构中的应用,帮助大家更全面地掌握指针的精髓。
在这里插入图片描述

一、数组名的理解

数组名在 C 语言中是一个非常特殊的概念。它本质上是一个指向数组首元素的指针。例如,对于数组 int arr[5];arr 就是数组名,它指向数组的第一个元素 arr[0]。我们可以通过打印 arr&arr[0] 的值来验证这一点,它们的值是相同的。

#include <stdio.h>int main() {int arr[5] = {1, 2, 3, 4, 5};printf("arr的地址:\t\t%p\n", arr);printf("数组首元素的地址:\t%p\n", &arr[0]);return 0;
}

数组名的理解

运行结果会显示两个地址相同,这证明了数组名就是一个指向首元素的指针。

数组名作为一个指针,它具有一些特殊的性质。首先,数组名的值是常量,不能被修改。也就是说,我们不能对数组名进行赋值操作,如 arr = &arr[1]; 是非法的。其次,数组名的类型是指向数组中元素类型的指针。例如,对于 int arr[5];arr 的类型是 int*,即指向整型的指针。

二、使用指针访问数组

既然数组名是首元素的地址,我们就可以通过指针来访问数组中的元素。例如,*(arr + i) 就相当于 arr[i],表示数组中第 i 个元素的值。

#include <stdio.h>int main() {int arr[5] = {1, 2, 3, 4, 5};for (int i = 0; i < 5; i++) {printf("arr[%d] = %d\n", i, *(arr + i));}return 0;
}

在这里插入图片描述

这段代码通过指针的方式遍历了数组中的每个元素,输出结果与直接使用数组下标访问相同。

指针访问数组元素的原理是基于指针的偏移。指针本身存储的是一个内存地址,当我们对指针进行加法运算时,实际上是根据指针所指向的数据类型计算偏移量。例如,对于 int* ptr;ptr + 1 的值是 ptr 的值加上 sizeof(int),即指针指向下一个整数元素的地址。

三、一维数组传参的本质

当我们将一维数组作为参数传递给函数时,实际上传递的是数组的首地址。也就是说,函数内部接收到的并不是整个数组,而是一个指向数组首元素的指针。因此,在函数内部,我们可以通过指针来操作数组中的元素。

#include <stdio.h>void printArray(int arr[], int size) {for (int i = 0; i < size; i++) {printf("%d ", arr[i]);}printf("\n");
}int main() {int arr[5] = {1, 2, 3, 4, 5};printArray(arr, 5);return 0;
}

在这里插入图片描述

在这个例子中,printArray 函数接收一个指针和数组大小作为参数,然后通过指针访问数组中的元素并打印出来。

需要注意的是,当数组作为函数参数传递时,数组的大小信息会丢失。因此,我们需要额外传递一个参数来指定数组的大小,以便在函数内部正确地操作数组。

四、冒泡排序

冒泡排序是一种简单的排序算法,它通过比较相邻元素的大小并交换位置来实现排序。在实现冒泡排序时,我们通常会使用指针来操作数组元素。

#include <stdio.h>void bubbleSort(int arr[], int size) {for (int i = 0; i < size - 1; i++) {for (int j = 0; j < size - 1 - i; j++) {if (arr[j] > arr[j + 1]) {int temp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = temp;}}}
}int main() {int arr[5] = {5, 3, 8, 1, 2};bubbleSort(arr, 5);printf("排序后的数组:");for (int i = 0; i < 5; i++) {printf("%d ", arr[i]);}return 0;
}

冒泡排序

这段代码实现了冒泡排序算法,通过指针操作数组元素,将数组从小到大排序。

冒泡排序的时间复杂度为 O(n²),其中 n 是数组的大小。虽然它的效率不高,但对于小规模的数据集或者学习排序算法的基本原理来说,冒泡排序是一个很好的起点。

五、二级指针

二级指针是指指向指针的指针。它在处理指针数组或者动态分配的二维数组时非常有用。例如,我们可以通过二级指针来操作一个字符串数组。

#include <stdio.h>int main() {char *strArray[] = {"Hello", "World", "C", "Language"};char **ptr = strArray;for (int i = 0; i < 4; i++) {printf("%s\n", *(ptr + i));}return 0;
}

二级指针

在这个例子中,strArray 是一个指针数组,ptr 是一个二级指针,指向 strArray 的首元素。通过 ptr 我们可以访问数组中的每个字符串。

二级指针的使用需要注意指针的初始化和解引用操作。在使用二级指针之前,必须确保它指向一个有效的内存地址,并且在解引用时要逐步进行,先获取一级指针,再访问最终的数据。

六、指针数组

指针数组是一个数组,其元素都是指针类型。常见的应用场景是存储多个字符串的首地址,方便对字符串进行操作。

#include <stdio.h>int main() {char *strArray[] = {"Apple", "Banana", "Cherry", "Date"};for (int i = 0; i < 4; i++) {printf("%s\n", strArray[i]);}return 0;
}

在这里插入图片描述

这里定义了一个指针数组 strArray,每个元素都是一个指向字符的指针,存储了不同水果名称的首地址。

指针数组的优势在于可以方便地对一组指针进行统一管理。例如,我们可以对指针数组中的字符串进行排序、查找等操作。

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

利用指针数组,我们可以模拟二维数组的操作。通过动态分配内存,我们可以根据需要创建不同大小的“二维数组”。

#include <stdio.h>
#include <stdlib.h>int main() {int rows = 3, cols = 4;int **array = (int **)malloc(rows * sizeof(int *));for (int i = 0; i < rows; i++) {array[i] = (int *)malloc(cols * sizeof(int));}// 初始化数组for (int i = 0; i < rows; i++) {for (int j = 0; j < cols; j++) {array[i][j] = i * cols + j + 1;}}// 打印数组for (int i = 0; i < rows; i++) {for (int j = 0; j < cols; j++) {printf("%d ", array[i][j]);}printf("\n");}// 释放内存for (int i = 0; i < rows; i++) {free(array[i]);}free(array);return 0;
}

在这里插入图片描述

这段代码通过指针数组模拟了一个 3 行 4 列的二维数组,实现了动态内存分配、初始化、打印和释放内存的操作。

在动态分配内存时,我们需要先为指针数组本身分配内存,然后再为每个指针元素分配相应的内存。这样可以灵活地控制每一行的大小,甚至可以创建不规则的二维数组。

总结

通过本文的学习,我们深入理解了指针在数组、函数传参以及复杂数据结构中的应用。从数组名作为指针的概念,到使用指针访问数组元素,再到一维数组传参的本质,我们逐步掌握了指针在数组操作中的核心技巧。同时,我们还学习了冒泡排序算法的实现,以及二级指针、指针数组和指针数组模拟二维数组的用法。这些知识将为我们进一步学习 C 语言中的高级指针应用打下坚实的基础。在实际编程中,灵活运用指针可以让我们更高效地操作数据,实现复杂的功能。希望同学们能够多加练习,通过实际编写代码来加深对指针的理解和掌握。

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

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

相关文章

高效内存管理:x86-64架构中的分页机制

在 x86-64 架构的世界里&#xff0c;内存分页机制扮演着举足轻重的角色&#xff0c;它就像是一座桥梁&#xff0c;连接着虚拟地址与物理地址。简单来说&#xff0c;内存分页机制就是将线性地址&#xff08;也就是虚拟地址&#xff09;切分成一个个固定大小的页&#xff0c;并把…

统一开放世界与开放词汇检测:YOLO-UniOW无需增量学习的高效通用开放世界目标检测框架

目录 一、摘要 二、引言 三、相关工作 开放词汇对象检测 开放世界目标检测 参数高效学习 四、高效通用的开放世界目标检测 问题定义 高效的自适应决策学习 开放世界通配符学习 五、Coovally AI模型训练与应用平台 六、实验 数据集 评价指标 实施细节 定量结果 …

fileinclude

##解题思路 场景首页没有什么提示&#xff0c;只有个flag在flag.php中&#xff0c;而且需要更改language&#xff0c;还有个index.php的路径&#xff0c;先记住它 习惯性查看源代码&#xff0c;得到了题目真正的内容&#xff0c;关键在于lan变量读取我们传入的Cookie值中的lang…

链表-LeetCode

这里写目录标题 1 排序链表1.1 插入法 O&#xff08;n&#xff09;1.2 归并排序 1 排序链表 1.1 插入法 O&#xff08;n&#xff09; /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullpt…

计算机网络基础:WiFi 与蓝牙的原理与应用

计算机网络基础:WiFi 与蓝牙的原理与应用 一、前言二、WiFi 原理2.1 概述2.2 工作频段2.2.1 2.4GHz 频段2.2.2 5GHz 频段2.3 调制技术2.3.1 正交频分复用(OFDM)2.3.2 直接序列扩频(DSSS)2.4 通信协议2.5 网络架构2.5.1 独立基本服务集(IBSS)2.5.2 基础服务集(BSS)2.5.…

深入解析 Java 类加载机制及双亲委派模型

&#x1f50d; Java的类加载机制是确保应用程序正确运行的基础&#xff0c;特别是双亲委派模型&#xff0c;它通过父类加载器逐层加载类&#xff0c;避免冲突和重复加载。但在某些特殊场景下&#xff0c;破坏双亲委派模型会带来意想不到的效果。本文将深入解析Java类加载机制、…

【数据可视化艺术·进阶篇】热力图探秘:用色彩演绎场馆和景区的人流奥秘

假期出游&#xff0c;你是不是也遇到过这样的状况&#xff1a;想去的热门景点&#xff0c;放眼望去全是攒动的人头&#xff0c;根本没法好好欣赏风景&#xff1b;而景区里一些小众角落&#xff0c;却冷冷清清&#xff0c;鲜有人至。还有在轨道交通枢纽、大型体育场这些地方&…

理解文字识别:一文读懂OCR商业化产品的算法逻辑

文字识别是一项“历久弥新”的技术。早在上世纪初&#xff0c;工程师们就开始尝试使用当时有限的硬件设备扫描并识别微缩胶片、纸张上的字符。随着时代和技术的发展&#xff0c;人们在日常生活中使用的电子设备不断更新换代&#xff0c;文字识别的需求成为一项必备的技术基础&a…

智能监控视频聚合平台,GB28181/RTSP/SIP/RTMP直播会议融合方案

全场景智能监控聚合平台&#xff1a;打破边界&#xff0c;赋能高效协同 在数字化转型加速的今天&#xff0c;海量视频监控设备、多样化的编码协议与复杂的业务场景&#xff0c;让企业面临跨系统整合难、资源调度效率低、协作响应慢等痛点。我们的智能监控聚合平台以技术创新为…

【机器学习】imagenet2012 数据预处理数据预处理

【机器学习】数据预处理 1. 下载/解压数据2. 数据预处理3. 加载以及训练代码3.1 使用PIL等加载代码3.2 使用OpenCV的方式来一张张加载代码3.3 h5的方式来加载大文件 最后总结 这个数据大约 140个G,128w的训练集 1. 下载/解压数据 首先需要下载数据&#xff1a; 数据最后处理…

语言模型理论基础-持续更新-思路清晰

1.预训练 相似的任务A、B&#xff0c;任务A已经用大数据完成了训练&#xff0c;得到模型A。 我们利用-特征提取模型的-“浅层参数通用”的特性&#xff0c;使用模型A的浅层参数&#xff0c;其他参数再通过任务B去训练&#xff08;微调&#xff09;。 2.统计语言模型 通过条件…

IDEA的基础快捷键

文章目录 1、书写main函数2、书写输出函数println3、书写for循环4、输出变量的值或者输出函数求的值5、代码注释7、主题、字体设置8、自动生成使用信息9、关闭启动IDEA默认打开上次的项目10、字体放大放小11、代码缩进12、快速复制/删除一行13、回退14、字母大小写转换15、调试…

音视频 二 看书的笔记 MediaPlayer

此类是用于播放声音和视频的主要 API 对方不想多说向你丢了一个链接 MediaPlayer Idle 空闲状态Initialized 初始化状态 调用 setDataSource() 时会进入此状态 setDataSource必须在Idle 状态下调用&#xff0c;否则就抛出异常了了了了了。Prepared 准备状态 回调监听setOnPrep…

Linux笔记---动静态库(使用篇)

目录 1. 库的概念 2. 静态库&#xff08;Static Libraries&#xff09; 2.1 静态库的制作 2.2 静态库的使用 2.2.1 显式指定库文件及头文件路径 2.2.2 将库文件安装到系统目录 2.2.3 将头文件安装到系统目录 3. 动态库 3.1 动态库的制作 3.2 动态库的使用 3.2.1 显式…

CAS(Compare And Swap)

CAS核心原理 操作流程 CAS 包含三个参数&#xff1a;内存值&#xff08;V&#xff09;、预期值&#xff08;E&#xff09;和新值&#xff08;N&#xff09;。执行步骤如下&#xff1a; 比较&#xff1a;检查当前内存值 V 是否等于预期值 E。 交换&#xff1a;如果相等&#…

宝塔面板安装docker flarum失败,请先安装依赖应用: [‘mysql‘]:5/8

安装失败的解决方案 提示错误请先安装依赖应用: [mysql]:5/8 解决方案&#xff1a;不要使用最新的docker mysql&#xff0c;使用5.7.44版本docker mysql&#xff0c;等安装完毕再安装docker flarum就不会报错了。 如果安装完成你不知道默认的账号密码可以看这里 宝塔docker f…

c#的.Net Framework 的console 项目找不到System.Window.Forms 引用

首先确保是建立的.Net Framework 的console 项目,然后天健reference 应用找不到System.Windows.Forms 引用 打开对应的csproj 文件 在第一个PropertyGroup下添加 <UseWindowsForms>true</UseWindowsForms> 然后在第一个ItemGroup 下添加 <Reference Incl…

基于 mxgraph 实现流程图

mxgraph 可以实现复杂的流程图绘制。mxGraph里的Graph指的是图论(Graph Theory)里的图而不是柱状图、饼图和甘特图等图(chart)&#xff0c;因此想找这些图的读者可以结束阅读了。 作为图论的图&#xff0c;它包含点和边&#xff0c;如下图所示。 交通图 横道图 架构图 mxGrap…

21.Excel自动化:如何使用 xlwings 进行编程

一 将Excel用作数据查看器 使用 xlwings 中的 view 函数。 1.导包 import datetime as dt import xlwings as xw import pandas as pd import numpy as np 2.view 函数 创建一个基于伪随机数的DataFrame&#xff0c;它有足够多的行&#xff0c;使得只有首尾几行会被显示。 df …

STL之空间配置器

1. 什么是空间配置器 空间配置器&#xff0c;顾名思义就是为各个容器高效的管理空间(空间的申请与回收)的&#xff0c;在默默地工作。虽然在常规使用STL时&#xff0c;可能用不到它&#xff0c;但站在学习研究的角度&#xff0c;学习它的实现原理对我们有很大的帮助。 2. 为什…