C语言——深入理解指针(第四章)

news/2024/4/25 12:32:40

一、二级指针

在讲二级指针之前,我们先回顾一下指针的定义一直之前讲的一级指针。

1.指针的定义

  • 一级指针:是一个指针变量,指向一个普通变量,并保存该普通变量的地址;
  • 二级指针:是一个指针变量,指向一个一级指针,并保存该一级指针的地址;

2.引入二级指针

由于一级指针已经很熟悉,这里不再赘述,这里我们只谈谈二级指针
下面先简单使用一个二级指针看看

#include <stdio.h>int main()
{int a = 10;int b = 20;int* p = &a;int** s = &p;//一次解引用*s 此时类型int**s = &b;//二次解引用**s 此时类型int**s = 200;return 0;
}

逻辑关系如下:

a是一个int类型的变量,一级指针p指向a,并保存a的地址;
二级指针变量s指向一级指针变量p,并保存p的地址。

二级指针s解引用操作:

  • 一次解引用
    *s的类型变成了(int*)(代表着一级指针p)间接改变了p的指向,从a的地址变成了b的地址;
  • 二次解引用
    s的类型变成了int (代表着变量b),此时s = 200;(等价于b = 200;)

3.深入理解二级指针

(1)下面举个例子:
#include <stdio.h>int main()
{//普通变量int a1 = 1;int a2 = 1;int a3 = 1;//一级指针int* p1 = &a1;int* p2 = &a2;int* p3 = &a3;//二级指针int** s = &p1;return 0;
}

(假设a1,a2,a3空间连续,p1,p2,p3空间连续)逻辑图如下: 

接下来我们结合上面的逻辑图看看下面这张表

 

分析:

s+1 表示二级指针s指向了p2,,移动的字节数需要根据指向的数据的空间大小进行计算sizeof(int*) * 1,所以移动4字节,此时s+1还是二级指针,所以类型int**
*s+1 先对s进行一次解引用为*s,相当于操控一级指针p1,然后*s + 1 ,相当于p1指向了a2的地址,所以移动了sizeof(int) * 1 = 4字节 ,此时的类型为 int*
**s + 1表示二次解引用,相当于a1的值+1,所以a1 = 2; a1的类型就是int

(2)在(1)的前提下,那要是把所有的类型换成char、short、double类型呢 

    char类型(1字节)

    short类型(2字节)

    double类型(8字节) 

4.总结 

在深入理解了二级指针的逻辑处理过程之后,我们不难发现:
对二级指针变量s指针的移动操作时,s都会将以保存的一级指针的类型进行解析步长(s + sizeof( p) * n),而一级指针*s(相当于p一级指针变量)会以保存的变量的类型进行解析步长(*s + sizeof(a) *n)

n指的是移动的次数,本例题是n = 1

二、指针数组

1.什么是指针数组

指针数组是指针还是数组?

我们类比⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。

那指针数组呢?是存放指针的数组。

指针数组的每个元素都是用来存放地址(指针)的。

如下图: 

指针数组的每个元素是地址,⼜可以指向⼀块区域。

指针数组,即数组中的每个元素都是一个指针。这些指针可以指向任何类型的数据。指针数组通常用于存储一组数据的地址,或传递一组数据的地址给函数。

2.指针数组的声明和初始化

下面是指针数组的声明方法:

type* arrayName[size];

其中 type 表示指针指向的数据类型,arrayName 是指针数组的名字,size 表示指针数组的大小。 

接下来是指针数组的初始化:

type *arrayName[size] = {pointer1, pointer2, pointer3, ... };

其中 pointer1pointer2pointer3 等表示指针数组中每个元素指向的地址。注意,指针数组也可以不进行初始化。 

下面是一个实际例子:

int a = 1, b = 2, c = 3, d = 4;
int *ptrArray[4] = {&a, &b, &c, &d};

上面的代码声明了一个指针数组,其中每个元素都是 int 型指针。然后分别把 abcd 的地址存储到 ptrArray 数组中。 

3.指针数组元素的访问与修改

访问指针数组的元素可以通过以下两种方式:

  1. arrayName[index]
  2. *(arrayName + index)

修改指针数组的元素同样可以通过以上两种方式。

下面是一个例子:

int a = 1, b = 2, c = 3, d = 4;
int *ptrArray[4] = {&a, &b, &c, &d};
printf("ptrArray[1] = %d\n", *ptrArray[1]); // 输出 2
*ptrArray[1] = 5;
printf("b = %d\n", b); //输出 5

上面的代码展示了如何访问和修改指针数组内的元素。

4.指针数组元素的地址获取 

通过使用 &*arrayName[index] 可以获得指针数组中某个元素的地址。

下面是一个例子:

int a = 1, b = 2, c = 3, d = 4;
int *ptrArray[4] = {&a, &b, &c, &d};
printf("&*ptrArray[1] = %p\n", &*ptrArray[1]); // 输出元素 1 的地址

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

#include <stdio.h>
int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 2,3,4,5,6 };int arr3[] = { 3,4,5,6,7 };//数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中int* parr[3] = { arr1, arr2, arr3 };int i = 0;int j = 0;for (i = 0; i < 3; i++){for (j = 0; j < 5; j++){printf("%d ", parr[i][j]);}printf("\n");}return 0;
}

画图解释 

 

parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数组中的元素。

上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并非是连续的。 


https://www.xjx100.cn/news/3091602.html

相关文章

轻松驾驭Linux命令:账户查看、目录文件操作详解

&#x1f3a5; 屿小夏 &#xff1a; 个人主页 &#x1f525;个人专栏 &#xff1a; Linux系统操作 &#x1f304; 莫道桑榆晚&#xff0c;为霞尚满天&#xff01; 文章目录 &#x1f4d1;引言&#x1f324;️查看账户☁️whoami☁️who &#x1f324;️ls和目录文件的创建删除☁…

Vue3入门(与Vue2进行对比)

1. Vue2 选项式 API vs Vue3 组合式API 特点&#xff1a; 代码量变少分散式维护变成集中式维护 2. Vue3的优势 使用create-vue搭建Vue3项目 1. 认识create-vue create-vue是Vue官方新的脚手架工具&#xff0c;底层切换到了 vite &#xff08;下一代前端工具链&#xff09;&…

axios 请求合集

post 请求 请求负载请求参数&#xff08;Request Payload&#xff09; import axios from axios import qs from query-stringexport function getRoles(data){return axios.post(目标地址,data,{headers:{Content-Type: application/json,},}) }表单请求参数&#xff08;Form…

鸿蒙原生应用/元服务开发-AGC分发如何配置版本信息(上)

1.配置HarmonyOS应用的“发布国家或地区”。 2.设置是否为开放式测试版本。 注意&#xff1a;HarmonyOS应用开放式测试当前仅支持手机、平板、智能手表。如开发者想发布为开放式测试版本&#xff0c;选择“是”。正式发布的版本请选择“否”。 3.在“软件版本”下点击“软件包…

MAX/MSP SDK学习03:Atoms and Messages的使用

今天终于把Message消息选择器看得有点头绪了&#xff0c;主要是这个官方英文文档理解起来有点抽象。 编写IsMatchABC自定义Object&#xff0c;要求&#xff1a; ①若左入口&#xff08;入口0&#xff09;收到 "int" 型消息&#xff0c;则从出口发送数值 "888&q…

计时器 Timer(Kotlin Flow)

代码&#xff1a; class FlowTimer(private val duration: Int,private val scope: CoroutineScope,private val onTick: (Int) -> Unit,private val onStart: (() -> Unit)? null,private val onFinish: (() -> Unit)? null,private val interval: Int 1 ) {p…

基于Java GUI模拟银行自动取款机

一、课题描述 编写一个程序&#xff0c;模拟银行自动取款机的工作流程。主要功能如下所述: 当输入给定的卡号和密码&#xff08;初始卡号为888888和密码为123456)时&#xff0c;系统能登录ATM柜员机系统,用户可以按照以下规则进行: 查询余额:初始余额为50000元 ATM取款:每次…

Python-函数传参与数据类型

Python中&#xff0c;函数参数传递是通过对象的引用进行的&#xff0c;我们可以进行下面的验证。 def use_name(val):print("name id :%s" % (id(val)))val "hanshu1"print("name id modified :%s" % (id(val)))def test_ref():name "ha…