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


    一、二级指针

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

    1.指针的定义

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

    2.引入二级指针

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

    1. #include
    2. int main()
    3. {
    4. int a = 10;
    5. int b = 20;
    6. int* p = &a;
    7. int** s = &p;
    8. //一次解引用*s 此时类型int*
    9. *s = &b;
    10. //二次解引用**s 此时类型int
    11. **s = 200;
    12. return 0;
    13. }

    逻辑关系如下:

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

    二级指针s解引用操作:

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

    3.深入理解二级指针

    (1)下面举个例子:
    1. #include
    2. int main()
    3. {
    4. //普通变量
    5. int a1 = 1;
    6. int a2 = 1;
    7. int a3 = 1;
    8. //一级指针
    9. int* p1 = &a1;
    10. int* p2 = &a2;
    11. int* p3 = &a3;
    12. //二级指针
    13. int** s = &p1;
    14. return 0;
    15. }

    (假设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 等表示指针数组中每个元素指向的地址。注意,指针数组也可以不进行初始化。 

    下面是一个实际例子:

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

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

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

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

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

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

    下面是一个例子:

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

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

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

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

    下面是一个例子:

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

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

    1. #include
    2. int main()
    3. {
    4. int arr1[] = { 1,2,3,4,5 };
    5. int arr2[] = { 2,3,4,5,6 };
    6. int arr3[] = { 3,4,5,6,7 };
    7. //数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中
    8. int* parr[3] = { arr1, arr2, arr3 };
    9. int i = 0;
    10. int j = 0;
    11. for (i = 0; i < 3; i++)
    12. {
    13. for (j = 0; j < 5; j++)
    14. {
    15. printf("%d ", parr[i][j]);
    16. }
    17. printf("\n");
    18. }
    19. return 0;
    20. }

    画图解释 

     

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

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

  • 相关阅读:
    react中hook 函数的使用
    恶意代码防范技术笔记(四)
    CPP语法(六)——函数模板
    centos6.5 安装xtrabackup,mysql备份工具
    【金九银十必问面试题】站在架构师角度分析问题,如何解决TCC中的悬挂问题
    字节开源RPC框架Kitex的日志库klog源码解读
    opencv视频文件的读取,处理与保存
    华为云云耀云服务器L实例评测|部署spring项目端口开放问题的解决 & 服务器项目环境搭建MySQL,Redis,Minio...指南
    使用定时器中断进行延时,可能会遇到的一个小bug
    网络请求——跨域 的概念
  • 原文地址:https://blog.csdn.net/Evan26/article/details/134478871