【C语言】带你彻底理解指针(详解)

简介: 指针的介绍:指针是一个用来存放地址的变量,可以通过指针存放的地址找到对应位置的值,对其进行使用。指针在32位平台下的大小是4个字节,在64位平台下是8个字节。(这是因为32位平台下内存地址是由32根地址线组成,一根地址线就是1bit,用4个字节就可以存下,而64位平台有64根地址线,需要八个字节才可以存下)。如果对指针进行加减与解引用操作,则±的步长与解引用操作的长度由指针类型决定,指针类型是几个字节±步长与解引用就是几个字节。

指针的介绍:

指针是一个用来存放地址的变量,可以通过指针存放的地址找到对应位置的值,对其进行使用。

指针在32位平台下的大小是4个字节,在64位平台下是8个字节。(这是因为32位平台下内存地址是由32根地址线组成,一根地址线就是1bit,用4个字节就可以存下,而64位平台有64根地址线,需要八个字节才可以存下)。

  1. 如果对指针进行加减与解引用操作,则±的步长与解引用操作的长度由指针类型决定,指针类型是几个字节±步长与解引用就是几个字节。

一、简单指针🌈

char* pc(字符指针)

int *pi(整形指针)

float* pf(浮点型指针)

还有short…double等等,拿部分举例。

1.1 指针的定义与使用

  • 这三种指针的用法基本相同,只不过char*指针存放的是char变量地址,int型指针存放的是int变量地址,float指针存放的是float变量地址。


#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
    char a = 'a';   //定义一个char类型变量
    char* pa = &a;     //*号先与pa结合说明pa是一个指针类型,指向的是char型数据的地址,将a的地址给到pa
    *pa = 'b';      //通过pa改变a的值
    printf("%c", a);  // b
    int num = 10;   
    int* pnum = &num;
    *pnum = 20;
    printf("%d", num);
    float f = 9.0;
    float* pf = &f;
    *pf = 5.0;
    printf("%d", f);
}

我们可以使用指针对指针指向的变量进行使用(修改或输出)。


1.2 指针与数组

  • 数组名就是数组的首地址。

代码示例


3307af313fe44087a82b33de549e48e3.png

  • 得到数组首地址可以通过指针遍历数组

代码示例


bb84ab14de204741a38b7385d2afcc2e.png

二、指针数组

指针数组就是用来装指针的数组

int p1[];   //(方便理解拿整形数组对比)p1与[]结合说明p1是数组,装的是int类型的数据
int* p2[];  //[]优先级比*高 p2先与[]结合,说明p2是一个数组 里面装的是int*类型的数据

举个例子

#include <stdio.h>
int main()
{
    int a = 0;
    int b = 0;
    int c = 0;
  int* arr[3] = {&a,&b,&c};   //指针数组就是数组的每个位置都存储一个地址(指针)
}

三、数组指针🌞

3.1 数组指针的定义

数组指针与指针数组名字非常相似但是却截然不同,数组指针是一个指针,指针数组是一个数组

//对比一下指针数组和数组指针
int* pa[];    //这是指针数组,优先级问题,pa先与[]结合,说明pa是一个数组,装的是int*的数据
int (*pa)[]   //这是数组指针,()优先级更高,pa先与*结合,说明pa是一个指针,指向一个int类型的数组

3.2 ”数组名“与”&数组名“

前面说了数组名代表数组首元素地址,那&数组名呢?

代码示例


304cd45628634b39b37672eafa23ca09.png

  • 对数组名与&数组名进行输出,发现地址一样


5dc3758e0aab4877ad45e2d5a613b391.png

  • 对数组名与&数组名+1,发现数组名加了四个字节,而&数组名加了十六进制的28,也就是十进制的40,为整个数组的大小。

结论:数组名与&数组名地址一样,但是数组名代表的是首元素地址,而&数组名代表的是整个数组。

3.3 数组指针使用

void print(int (*pa)[3])
{
    for(int i=0;i<2;i++)
    {
    for(int j=0;j<3;j++)
        {
      printf("%d",arr[i][j]);
        }
    }
}
int main()
{
  int arr[2][3] = {0,1,2,3,4,5};
    print(arr);   //二维数组首地址相当于 &(第一个一维数组)
}

四、函数指针🤓

函数指针就是指向函数的指针

  • 函数指针的定义
  • *函数返回值 (pa)(函数参数)

可以看作pa先与*结合,说明pa是个指针,指向一个函数

4.1函数指针的使用

代码示例

int add(int a,int b)
{
  return a+b;
}
int main()
{
  int (*pa)(int ,int) = add;
    int a = 3;
    int b = 4;
    printf("%d",pa(3,4));   //7
}

五、函数指针数组🚀

函数指针数组就是用来装函数指针的数组,需要注意每个函数指针数组的返回值和参数必须相同类型

  • 函数指针数组的定义
  • *函数返回值 (pa[])(函数参数)

可以看作pa先与[]结合说明pa是一个数组,每个位置装的是函数指针(地址)

5.1 函数指针数组的使用

#include <stdio.h>
int add(int a, int b) //需要注意函数指针数组的每个参数类型与返回值类型必须相同
{
  return a + b;
}
int sub(int a, int b)
{
  return a - b;
}
int mul(int a, int b)
{
  return a * b;
}
int div(int a, int b)
{
  return a / b;
}
void menu()
{
  printf("***************\n");
  printf(" 1:add 2:sub \n");
  printf(" 3:mul 4:div \n");
  printf("    0:quit   \n");
  printf("**************\n");
}
int main()
{
  int x = 0;
  int y = 0;
  int input = 0;
  int ret = 0;
  int(*p[5])(int x, int y) = { NULL, add, sub, mul, div };
  do
  {
    menu();
    printf("请选择:");
    scanf("%d", &input);
    if (input >= 1 && input <= 4)
    {
      printf("输入两个数字:");
      scanf("%d %d", &x, &y);
      ret = (*p[input])(x, y);
    }
    else if (input == 0)
    {
      break;
    }
    else
    {
      printf("请输入0-4的数字\n");
      continue;
    }
    printf("%d\n", ret);
  } while (input);
}

bfbae4e1c1f94bb5bc7a630f8ed760b3.png

六、指向函数指针数组的指针🚀🚀

函数指针数组的定义为 int (*pa[5])(参数),指向函数指针数组的指针就是:

int (( * pa)[5]) (参数) //用括号把pa括起来使得pa先与 * 结合,说明pa是一个指针,返回值类型就为函数指针数组


61fbcd3a31a44aee892e1cc1db53d6c7.png

指向函数指针数组的指针的定义

void test(char* str)
{
  printf("指针真简单,%s",str);
}
int main()
{
    //定义一个函数指针数组  
    void (*pa[5])(char*) ;
    pa[0] = test;
    //指向函数指针数组的指针
    void (*(*ppa)[5])(char*) = &pa;
    return 0;
}

七、回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应 。

  • stdlib头文件下的qsort()函数就是一个回调函数,链接: qsort函数详解


完结

创作不易,还请各位小伙伴多多点赞👍关注✨收藏⭐



d89241b7267e47efa46e331647af0e19.gif
















相关文章
|
3月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
68 0
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
106 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
89 9
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
53 7
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
222 13
|
2月前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
78 11
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
180 3
|
2月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
2月前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
57 1

热门文章

最新文章