C语言指针与数组的深度应用与内存解析

C语言指针与数组的深度应用与内存解析

C语言指针与数组的深度应用与内存解析

在这里插入图片描述

💡 学习目标:掌握指针与数组的等价性原理,熟练运用指针操作数组元素,理解二者在内存中的存储本质,解决实际开发中数组遍历、数据拷贝的高效实现问题。
💡 学习重点:指针与数组名的区别、指针算术运算操作数组、二维数组的指针访问方式、内存视角下的数组与指针关系。

48.1 指针与数组的核心关联:本质与等价性

在C语言中,指针和数组的关系密不可分。很多初学者会混淆数组名和指针的概念,实际上二者既有联系又有本质区别。

48.1.1 数组名的“隐式转换”特性

当数组名出现在表达式中时,它会隐式转换为指向数组首元素的指针。我们可以通过一个简单的例子来验证这个特性:

#include<stdio.h>intmain(){int arr[5]={10,20,30,40,50};// 输出数组首元素地址printf("数组名arr的地址:%p\n", arr);// 输出数组首元素的指针地址printf("&arr[0]的地址:%p\n",&arr[0]);// 用数组名访问首元素printf("arr[0] = %d\n", arr[0]);// 用指针方式访问首元素printf("*arr = %d\n",*arr);return0;}

运行结果

数组名arr的地址:0x7ffeefbff560 &arr[0]的地址:0x7ffeefbff560 arr[0] = 10 *arr = 10 

从结果可以看出,arr&arr[0] 指向的是同一块内存地址。这就是数组名的隐式转换特性。

⚠️ 注意事项

  1. 数组名不是真正的指针变量,它是一个地址常量,不能被修改。比如 arr++ 这种写法是非法的。
  2. 当数组名作为 sizeof 操作符的参数时,不会发生隐式转换,此时 sizeof(arr) 计算的是整个数组的内存大小。

48.1.2 指针算术运算操作数组元素

指针的算术运算(加减整数)是操作数组的核心技巧。指针每加1,偏移的字节数等于其指向数据类型的大小。
我们可以用指针替代数组下标,实现更高效的数组遍历:

#include<stdio.h>intmain(){int arr[5]={10,20,30,40,50};// 定义指针指向数组首元素int*p = arr;int i;for(i =0; i <5; i++){// 指针算术运算:p+i 指向第i个元素printf("arr[%d] = %d,*(p+%d) = %d\n", i, arr[i], i,*(p+i));}return0;}

运行结果

arr[0] = 10,*(p+0) = 10 arr[1] = 20,*(p+1) = 20 arr[2] = 30,*(p+2) = 30 arr[3] = 40,*(p+3) = 40 arr[4] = 50,*(p+4) = 50 

从代码中可以得出结论:arr[i] 等价于 *(arr+i),也等价于 *(p+i)

💡 性能技巧:在循环遍历数组时,使用指针操作的效率略高于下标操作。因为下标操作需要计算 arr+i 的地址,而指针可以直接通过自增实现偏移。

48.2 二维数组的指针访问:多维数组的内存布局

二维数组是C语言开发中处理表格数据的常用结构。理解二维数组的内存布局,才能用指针灵活操作它。

48.2.1 二维数组的内存本质

二维数组在内存中是连续存储的,不存在“行”和“列”的物理分隔。比如 int arr[2][3] = {{1,2,3},{4,5,6}},它的内存布局是:

地址偏移 → 0 4 8 12 16 20 元素值 → 1 2 3 4 5 6 

其中每个 int 类型占4个字节,所以相邻元素的地址偏移量为4。

二维数组的数组名 arr 可以看作是指向一维数组的指针arr 指向第一行的一维数组 arr[0]arr+1 指向第二行的一维数组 arr[1]

48.2.2 用指针访问二维数组元素

我们可以通过三种方式访问二维数组的元素:下标法、数组名指针法、普通指针法。

#include<stdio.h>intmain(){int arr[2][3]={{1,2,3},{4,5,6}};// 方式1:下标法printf("下标法访问:arr[1][2] = %d\n", arr[1][2]);// 方式2:数组名指针法:arr[i][j] 等价于 *(*(arr+i)+j)printf("数组名指针法:*(*(arr+1)+2) = %d\n",*(*(arr+1)+2));// 方式3:普通指针法:将二维数组看作一维数组int*p =&arr[0][0];printf("普通指针法:*(p+5) = %d\n",*(p+5));return0;}

运行结果

下标法访问:arr[1][2] = 6 数组名指针法:*(*(arr+1)+2) = 6 普通指针法:*(p+5) = 6 

⚠️ 注意事项

  1. 二维数组的指针 arr 是指向一维数组的指针,其类型为 int (*)[3],不能直接赋值给 int * 类型的指针。
  2. 只有当普通指针指向二维数组的首元素地址时,才能用 *(p+i) 的方式访问元素。

48.3 实战案例:指针实现数组的高效拷贝与逆序

掌握指针与数组的关联后,我们可以用指针实现更高效的数组操作。下面是两个实际开发中常用的案例。

48.3.1 案例1:指针实现数组拷贝函数

要求:编写一个函数,用指针将源数组的内容拷贝到目标数组,要求不使用下标。

#include<stdio.h>#include<string.h>// 数组拷贝函数:src源数组,dest目标数组,len数组长度voidarr_copy(int*src,int*dest,int len){int*p_src = src;int*p_dest = dest;// 循环拷贝每个元素while(len--){*p_dest++=*p_src++;}}intmain(){int src_arr[5]={1,2,3,4,5};int dest_arr[5]={0};int i;// 调用拷贝函数arr_copy(src_arr, dest_arr,5);// 打印目标数组printf("拷贝后的目标数组:");for(i =0; i <5; i++){printf("%d ", dest_arr[i]);}return0;}

运行结果

拷贝后的目标数组:1 2 3 4 5 

💡 技巧解析:函数中使用 *p_dest++ = *p_src++,先执行赋值操作,再将两个指针自增,实现了简洁高效的拷贝。

48.3.2 案例2:指针实现数组逆序

要求:编写一个函数,用指针将数组元素逆序排列,要求不使用额外数组空间。

#include<stdio.h>// 数组逆序函数:arr待逆序数组,len数组长度voidarr_reverse(int*arr,int len){// 指向数组首元素int*start = arr;// 指向数组尾元素int*end = arr + len -1;int temp;// 首尾交换,直到指针相遇while(start < end){ temp =*start;*start =*end;*end = temp; start++; end--;}}intmain(){int arr[5]={1,2,3,4,5};int i;printf("逆序前的数组:");for(i =0; i <5; i++){printf("%d ", arr[i]);}// 调用逆序函数arr_reverse(arr,5);printf("\n逆序后的数组:");for(i =0; i <5; i++){printf("%d ", arr[i]);}return0;}

运行结果

逆序前的数组:1 2 3 4 5 逆序后的数组:5 4 3 2 1 

💡 技巧解析:通过首尾两个指针向中间移动,交换对应位置的元素,空间复杂度为O(1),是最优的数组逆序实现方式。

48.4 内存视角的总结:指针与数组的核心区别

很多开发者会混淆指针和数组,我们从内存角度总结二者的核心区别:

对比维度数组名指针变量
内存属性地址常量,不能修改指向变量,可以修改指向
sizeof计算计算整个数组的内存大小计算指针本身的大小(32位系统4字节,64位系统8字节)
初始化方式int arr[5] = {1,2,3,4,5}int *p = arr

本章核心结论

  1. 数组名在表达式中会隐式转换为指向首元素的指针,但它不是真正的指针变量。
  2. arr[i] 等价于 *(arr+i),指针算术运算是操作数组的高效方式。
  3. 二维数组在内存中连续存储,可通过指向一维数组的指针或普通指针访问。

Read more

一款数据库SQL防火墙:可以拦截99.99%,可以阻止恶意SQL

一款数据库SQL防火墙:可以拦截99.99%,可以阻止恶意SQL

文章目录 * 一、SQL注入:那个偷偷溜进房子的"不速之客" * 二、三种模式,给数据库装上"智能门禁系统" * 三、又快又准又简单,这才是理想中的安全防护 * 1. 99.99%的拦截准确率,近乎"零误报" * 2. 性能损耗低于6%,业务无感 * 3. 两步配置,小白也能轻松上手 * 四、从党政到能源,为什么他们都选择了金仓? 在数字化转型的浪潮中,数据已成为企业的核心资产。然而,SQL注入攻击如同潜伏在阴影中的"不速之客",时刻威胁着数据库的安全。即使开发团队严守预编译、输入过滤等防线,遗留代码、第三方组件的漏洞或人为疏忽仍可能给攻击者可乘之机。难道只能被动挨打、疲于补漏吗?金仓数据库(KingbaseES)内置的SQL防火墙,

By Ne0inhk
二、Kafka核心架构与分布式存储

二、Kafka核心架构与分布式存储

思维导图 一、Kafka定位与核心特性 Kafka不仅是传统的消息队列中间件,更被官方定义为新一代的分布式事件流平台。它在海量流式计算场景中占据绝对核心地位,具备以下底层物理特性: 高吞吐与高并发:摒弃缓慢的随机寻址,深度依赖操作系统的页缓存与磁盘的顺序追加写。单机即可支撑每秒百万级的高并发数据吞吐。 可靠性与持久化存储:流动的数据直接落盘持久化至日志文件。配合多副本冗余机制,确保物理节点宕机时核心业务数据绝对不丢失。 高可扩展性与解耦:支持零停机数据处理。支持在线动态扩容Broker节点,自动实现海量数据流的负载均衡。极大解耦了微服务系统,提升了全链路数据处理效率。 二、分布式存储基石:HDFS架构深度剖析 要理解现代中间件的数据分布逻辑,必须先解剖大数据存储基石HDFS的底层架构。 HDFS采用中心化控制模型,由主管元数据的NameNode与负责物理存储的DataNode构成。一个超大文件会被物理切分为默认128MB的数据块,分散存储在不同DataNode的磁盘上。 为保障极高的容错率,HDFS制定了基于机架感知的副本放置关键原则。 默认的三副

By Ne0inhk
C语言Web开发:CGI、FastCGI、Nginx深度解析

C语言Web开发:CGI、FastCGI、Nginx深度解析

C语言Web开发:CGI、FastCGI、Nginx深度解析 一、前言:为什么Web开发是C语言开发的重要技能? 学习目标 * 理解Web开发的本质:编写程序实现Web应用、服务器端逻辑和客户端交互 * 明确Web开发的重要性:支撑互联网、电子商务、社交网络等领域的发展 * 掌握本章学习重点:CGI、FastCGI、Nginx的开发方法、避坑指南、实战案例分析 * 学会使用C语言开发Web应用,实现服务器端逻辑和客户端交互 重点提示 💡 Web开发是C语言开发的重要技能!随着互联网的普及,Web开发的需求越来越大,C语言的高性能和可移植性使其在Web开发中具有重要地位。 二、模块1:CGI(通用网关接口)基础 2.1 学习目标 * 理解CGI的本质:通用网关接口,用于Web服务器与服务器端程序之间的通信 * 掌握CGI的核心架构:Web服务器、CGI程序、客户端 * 掌握CGI的开发方法:使用C语言编写CGI程序 * 掌握CGI的避坑指南:避免环境变量未设置、避免输出格式错误、避免资源泄漏 * 避开CGI使用的3大常见坑

By Ne0inhk
Spring Boot 安全认证与授权

Spring Boot 安全认证与授权

Spring Boot 安全认证与授权 22.1 学习目标与重点提示 学习目标:掌握Spring Boot安全认证与授权的核心概念与使用方法,包括Spring Security的定义与特点、Spring Boot与Spring Security的集成、Spring Boot与Spring Security的配置、Spring Boot与Spring Security的认证、Spring Boot与Spring Security的授权、Spring Boot与Spring Security的实际应用场景,学会在实际开发中处理安全认证与授权问题。 重点:Spring Security的定义与特点、Spring Boot与Spring Security的集成、Spring Boot与Spring Security的配置、Spring Boot与Spring Security的认证、Spring Boot与Spring Security的授权、Spring Boot与Spring Security的实际应用场景。 22.2 Spring Security概述 Spring

By Ne0inhk