C语言复习概要(六)

news2025/1/14 18:02:22

在这里插入图片描述

公主请阅

    • 1. 深入理解数组与指针在C语言中的应用
      • 1.1 数组名的理解
    • 2. 使用指针访问数组
    • 3. 一维数组传参的本质
    • 4. 冒泡排序的实现
    • 5. 二级指针
    • 6. 指针数组
    • 7. 指针数组模拟二维数组
    • 8.总结

1. 深入理解数组与指针在C语言中的应用

数组与指针是C语言的核心概念之一,理解它们的本质及其相互关系是成为C语言高手的必经之路。数组为我们提供了一种存储大量相同类型数据的高效方式,而指针则为操作这些数据提供了极大的灵活性。本文将从数组名与指针的基本理解出发,深入探讨指针与数组的结合使用,并通过经典的冒泡排序算法和多级指针操作举例说明。

1.1 数组名的理解

数组是存储相同类型数据的线性结构。一个简单的一维数组的声明如下:

int arr[5] = {1, 2, 3, 4, 5};

在这个例子中,arr是一个包含5个int类型元素的数组。但是在更底层的C语言内存模型中,数组名实际上是一个常量指针,它指向数组的第一个元素的地址。也就是说,arr的值是&arr[0]。但需要注意的是,数组名本身是一个常量,不能像普通指针一样被重新赋值。

例如,下面的代码将会报错:

arr = &x; // 错误:数组名是常量,不能被赋值

这一点解释了为什么我们可以通过指针的方式来访问数组的元素:

int *p = arr;

在这个例子中,p指向arr[0],也就是数组的第一个元素。通过指针p,我们可以轻松访问整个数组元素:

for (int i = 0; i < 5; i++) {
    printf("%d ", *(p + i));  // 输出:1 2 3 4 5
}

这说明数组名其实是一种特殊的指针,它和指针的行为非常相似,但在某些情况下,数组名与普通指针的行为略有不同。

2. 使用指针访问数组

使用指针访问数组是C语言中非常常见的操作,它可以提高程序的效率。我们可以通过指针遍历数组,而不是通过索引。

假设有一个数组:

int arr[5] = {10, 20, 30, 40, 50};

你可以通过指针访问数组的每一个元素:

int *ptr = arr;

for (int i = 0; i < 5; i++) {
    printf("Element %d: %d\n", i, *(ptr + i));
}

这里,ptr + i表示指向数组中第i个元素的指针,而*(ptr + i)则解引用该指针,得到对应的值。通过指针直接访问数组元素,避免了使用数组下标的额外计算,从而提升了访问速度。

3. 一维数组传参的本质

在C语言中,函数传递数组参数实际上是传递指针。让我们来看看一个示例:

void printArray(int *arr, int size) {
    for (int i = 0; i < size; i++) {
        printf("%d ", arr[i]);
    }
}

在这里,arr实际上是一个指向数组的指针,而不是整个数组。当我们调用这个函数时,传递的实际上是数组的地址:

int arr[5] = {1, 2, 3, 4, 5};
printArray(arr, 5);

这种方式使得数组传参非常高效,因为只需传递数组的首地址即可,而不是整个数组的内容。这样可以节省大量的内存和时间,特别是在处理大型数组时。

4. 冒泡排序的实现

冒泡排序是一种经典的排序算法,借助数组和指针的结合,我们可以用更灵活的方式实现它。让我们看看如何用指针来实现冒泡排序:

void bubbleSort(int *arr, int size) {
    for (int i = 0; i < size - 1; i++) {
        for (int j = 0; j < size - i - 1; j++) {
            if (*(arr + j) > *(arr + j + 1)) {
                // 交换两个元素
                int temp = *(arr + j);
                *(arr + j) = *(arr + j + 1);
                *(arr + j + 1) = temp;
            }
        }
    }
}

在这个例子中,我们通过指针访问数组中的元素并进行交换。与直接使用数组下标相比,使用指针能更加灵活地操作内存地址。

5. 二级指针

二级指针(即指向指针的指针)是C语言中的另一个重要概念。它常用于动态分配二维数组或在函数中修改指针的值。假设我们需要通过函数来修改一个指针的指向:

void modifyPointer(int **p) {
    static int x = 10;
    *p = &x;
}

在这个例子中,p是一个二级指针,它指向一个指针。通过解引用p,我们可以改变一级指针的值。在函数调用中:

int *ptr = NULL;
modifyPointer(&ptr);
printf("%d\n", *ptr);  // 输出:10

这展示了如何通过二级指针在函数中修改一级指针的值。

6. 指针数组

指针数组是一个存储指针的数组。它通常用于存储一组字符串或指向其他数组的指针。让我们看看一个指针数组的例子:

char *names[] = {"Alice", "Bob", "Charlie", "David"};

for (int i = 0; i < 4; i++) {
    printf("%s\n", names[i]);
}

在这个例子中,names是一个指针数组,每个元素都是一个指向字符串的指针。通过遍历指针数组,我们可以输出每个字符串。

指针数组在处理动态内存分配和多维数组时非常有用。

7. 指针数组模拟二维数组

二维数组是一种常见的数据结构,但有时候我们需要使用指针数组来模拟二维数组,以获得更大的灵活性。例如,以下代码演示了如何使用指针数组来动态创建一个二维数组:

int rows = 3, cols = 4;
int **arr = (int **)malloc(rows * sizeof(int *));
for (int i = 0; i < rows; i++) {
    arr[i] = (int *)malloc(cols * sizeof(int));
}

// 为数组赋值
for (int i = 0; i < rows; i++) {
    for (int j = 0; j < cols; j++) {
        arr[i][j] = i * cols + j;
    }
}

// 打印数组
for (int i = 0; i < rows; i++) {
    for (int j = 0; j < cols; j++) {
        printf("%d ", arr[i][j]);
    }
    printf("\n");
}

// 释放内存
for (int i = 0; i < rows; i++) {
    free(arr[i]);
}
free(arr);

这里,我们首先分配了一个存储行指针的数组,然后为每一行分配内存。这种方法使得我们可以根据需要动态调整二维数组的大小。

8.总结

数组与指针是C语言中的基础概念,但它们的结合使用可以极大提高程序的灵活性和效率。通过理解数组名的本质、指针访问数组、函数传参中的指针使用、以及指针数组与多级指针的应用,我们可以编写出高效且灵活的C程序。冒泡排序和二维数组的指针模拟为我们展示了指针在实际算法和数据结构中的应用。掌握这些技巧和概念,将帮助你在C语言编程中走得更远。

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

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

相关文章

无极低码课程【mysql windows下安装要点】

在Windows环境中安装MySQL 5.7教程 MySQL 是世界上最流行的开源关系型数据库管理系统之一。本教程将指导您在Windows操作系统上安装MySQL 5.7。 网上教程较多&#xff0c;这里不再详述&#xff0c;注意关键点即可 准备工作 下载MySQL 5.7安装包 访问 MySQL官方网站 下载MyS…

新年好——Dijkstra+Permutation

题目 代码 #include <bits/stdc.h> using namespace std; #define x first #define y second typedef pair<int, int> PII; const int N 5e410, M 2e510; const int inf 0x3f3f3f3f; int n, m; int a[6]; int h[N], e[M], ne[M], idx, w[M]; int dist[6][N];…

用Python实现运筹学——Day 17: 0-1 整数规划

一、学习内容 1. 0-1 整数规划的定义 0-1 整数规划是一类特殊的整数规划问题&#xff0c;决策变量只能取 0 或 1。它常用于解决选择问题&#xff0c;如是否选择某个项目、是否执行某个任务等。决策变量 ​ 通常表示“选择”&#xff08;&#xff09;或“不选择”&#xff08;…

实用宝典:元器件外贸独立站电子元件数据库设置完全手册

对于投身于元器件外贸领域的企业来说&#xff0c;如何建立一个既能凸显自身特色又具备高度功能性与良好用户体验的独立站&#xff1f;而在这一过程中&#xff0c;#电子元件数据库#作为独立站的核心要素之一&#xff0c;它的构建质量和管理方式又将如何直接影响网站的整体竞争力…

Python| 如何使用 DALL·E 和 OpenAI API 生成图像(2)

引言 想象一下&#xff0c;只要描述你想要的画面&#xff0c;电脑就能帮你画出来。这在几年前还像是科幻小说里的场景&#xff0c;但随着神经网络和潜在扩散模型&#xff08;LDM&#xff09;技术的发展&#xff0c;现在已经成为可能。OpenAI 推出的 DALLE 工具&#xff0c;因其…

STM32L151 多通道ADC DMA循环扫描STM32CubeIDE STM32CubeMX参考设计

简介 项目开发过程中&#xff0c;采用STM32L151 芯片进行涉及&#xff0c;其中需要使用其片上ADC进行多路ADC数据采样。这里就记录一下实际这块的开发过程&#xff0c;其中涉及工程代码再项目中实际投产使用。STM32L151 多通道ADC DMA循环扫描STM32CubeIDE STM32CubeMX参考设计…

【文心智能体 | AI大师工坊】通过知识库优化智能体『万圣节之纸人还魂』:探索恐怖剧本杀的奇幻之旅

文章目录 1.1、智能体运行效果1.2、创作灵感来源1.3、如何制作智能体1.4、可能会遇到的几个问题1.5、快速调优指南 My_优质智能体『万圣节之纸人还魂&#x1f47b;』&#xff1a;https://aq58pt.smartapps.baidu.com/?_swebfr1&_swebScene3611000100000000 在当今人工智能…

Vue开发中由错误These relative modules were not found 引起的问题思考及解决

一、由Vue: These relative modules were not found 引起的问题 1、vue2.6.14 These relative modules were not found 在使用vue2.6.14开发的项目在本地windows 10上都一直成功&#xff0c;想放到jenkins上进行发布。之前其它的 vue 项目也都能发布正常&#xff0c;但此次一…

【stm32】DMA的介绍与使用

DMA的介绍与使用 1、DMA简介2、存储器映像3、DMA框图4、DMA基本结构5、DMA请求6、数据宽度与对齐7、数据转运DMA&#xff08;存储器到存储器的数据转运&#xff09;程序编写&#xff1a; 8、ADC连续扫描模式DMA循环转运DMA配置&#xff1a;程序编写&#xff1a; 1、DMA简介 DM…

Antsword-labs靶机渗透

Less-1 在当前界面开启终端 ![](https://img-blog.csdnimg.cn/img_convert/e5ab1b947b1186a43b58abaf10263cb1.png) 启动环境 plain docker-compose up -d 蚁剑连接 ![](https://img-blog.csdnimg.cn/img_convert/81a5c09987e18355335d07e4da52cb5f.png) 打开终端寻找flag …

ACL:访问控制列表

基本概念 1.访问控制: 在路由器的入或者出的接口上&#xff0c;匹配流量&#xff0c;之后产生动作&#xff0c;只有允许和拒绝。 2.定义感兴趣流量: 帮助其他策略抓流量的 匹配规则:至上而下 逐一匹配 上条匹配按照上条执行 不再查看下条 (思科体系中 末尾隐含拒绝所有 华为…

20个月投标战胜国际对手,中国百余台AGV进驻欧洲……

导语 大家好&#xff0c;我是社长&#xff0c;老K。专注分享智能制造和智能仓储物流等内容。 在全球智能制造竞争日益激烈的今天&#xff0c;中国制造再次传来振奋人心的好消息。 经过长达20个月的艰苦角逐&#xff0c;一家中国机器人企业&#xff08;新松机器人&#xff09;成…

microsoft edge浏览器卡死问题

win11经常遇到microsoft edge浏览器卡死的情况&#xff0c;有时候是一会没用浏览器就全部卡死&#xff0c;有时候是锁屏或者电脑休眠浏览器就不能用&#xff0c;找了很多的办法都没好使&#xff0c;用以下方法好使了&#xff1a; edge浏览器中打开 edge://settings/system 把 …

计算机的错误计算(一百二十三)

摘要 探讨算式 的计算精度问题。 例1. 已知 计算 不妨在Python下计算&#xff0c;则有&#xff1a; 若用Rust在线计算&#xff1a; fn main() {let x: f64 0.125e-6;let tan_x x.tan();let sin_x x.sin();let denominator x - (1.0 x * x).sqrt();let result (…

大学模拟电路设计期末速成总结

大学模拟电路设计期末速成总结 模拟电路设计是电子工程领域的基础&#xff0c;它涉及到连续信号的处理和放大。对于电子工程的学生来说&#xff0c;掌握模拟电路设计的基本原理和应用是至关重要的。以下是期末速成总结&#xff0c;帮助你快速回顾和掌握模拟电路设计的关键知识…

香橙派刷机和开发环境准备(ubuntu22.04版)_随记1

前言&#xff1a; 一、香橙派刷ubuntu系统和SSH登录 1、刷机前准备&#xff1a; ①TF卡&#xff08;8G&#xff09;、读卡器、OrangePi5Pro ②Win32DiskImager&#xff08;烧写系统工具&#xff09;、SDFormatter&#xff08;TF格式化工具&#xff09; ③系统镜像&#xff…

学习文档(4)

目录 Vue简介 MVVM思想 Vue指令 内容输出指令 条件渲染指令 列表渲染指令 数据绑定指令 Vue简介 Vue2.x官网&#xff1a;https://v2.cn.vuejs.org Vue3.x官网&#xff1a;https://cn.vuejs.org 官网&#xff08;2.x版本&#xff09;对vue的定义是&#xff1a;vue是一套…

kubelet PLEG实现

概述 kubelet的主要作用是确保pod状态和podspec保持一致&#xff0c;这里的pod状态包括pod中的container状态&#xff0c;个数等。 为了达到这个目的&#xff0c;kubelet需要从多个来源watch pod spec的变化&#xff0c;并周期从container runtime获取最新的container状态。比如…

I2C相关结构体讲解:i2c_adapter、i2c_algorithm、i2c_msg

往期内容 I2C子系统专栏&#xff1a; I2C&#xff08;IIC&#xff09;协议讲解SMBus 协议详解 总线和设备树专栏&#xff1a; 专栏地址导航篇 – 专栏未篇 1.框图 建议右击图片在新标签页打开预览 i2c_transfer函数就是读取i2c设备的信息或者输出信息给i2c设备的函数 比如发送…

信息安全工程师(49)网络物理隔离系统与类型

前言 网络物理隔离系统是指通过物理隔离技术&#xff0c;在不同的网络安全区域之间建立一个能够实现物理隔离、信息交换和可信控制的系统&#xff0c;以满足不同安全域的信息或数据交换需求。 一、网络物理隔离系统概述 网络物理隔离系统的核心在于通过物理方式将不同安全级别的…