C语言:指针(超深度讲解)

简介: C语言:指针(超深度讲解)



学习目标:

1. 字符指针

2. 指针数组

3. 数组指针

4. 数组传参和指针传参

5. 函数指针

6. 函数指针数组

7. 指向函数指针数组的指针

8. 回调函数

指针:

指针可以理解为:

字符指针:

       定义:字符指针 char*。

字符指针的使用:

//使用1

int main ()

{

char ch = 'w' ;

char * pc = & ch ;

* pc = 'w' ;

return 0 ;

}

//使用2

int main ()

{

const char* pstr = "hello bit." ;//把一个常量字符串的 首字符 h 的地址 存放到指针变量 pstr

printf ( "%s\n" , pstr );

return 0 ;

}

练习:

指针数组:

       概念:指针数组是一个存放指针的数组。

int* arr1 [ 10 ];   // 整形指针的数组

char * arr2 [ 4 ];   // 一级字符指针的数组

char ** arr3 [ 5 ]; // 二级字符指针的数组

实现模拟二维数组:

数组指针:

       概念:能够指向数组的指针。(可以理解为先与指针结合再与数组结合)

     int (*p)[10];

// 解释: p先和*结合,说明p是一个指针变量 ,然后指着指向的是一个大小为 10 个整型的数组。所以 p 是一个指针,指 向一个数组,叫数组指针。

// 这里要注意: [ ] 的优先级要高于 * 号的,所以必须加上()来保证 p 先和 * 结合。

值得注意的是:

数组名的理解:数组名是数组首元素的地址
有2个例外:
1. sizeof(数组名),这里的数组名不是数组首元素的地址,数组名表示整个数组,sizeof(数组名)计算的是整个数组的大小,单位是字节
2. &数组名,这里的数组名表示整个数组, &数组名取出的是整个数组的地址
除此之外,所有的地方的数组名都是数组首元素的地址

数组指针一般用于二维数组:

数组的传参:

       二维数组的每一行可以理解为二维数组的一个元素每一行又是一个一维数组,所以二维数组其实是一维数组的数组。

       二维数组的数组名,也是数组名,数组名就是数组首元素的地址。

arr----首元素的地址;

arr----第一行的地址;
arr----一维数组的地址即数组的地址。

一维数组传参:

二维数组的传参:

总结:二维数组传参,函数形参的设计只能省略第一个[ ]的数字因为对一个二维数组可以不知道有多少行,但是必须知道一行多少元素这样才方便运算。

指针的传参:

一级指针传参:

二级指针的传参:

函数指针:

       概念:指向函数的指针。

  int (*pf)(int, int) = &Add;

   //pf是函数指针变量

   //int (*)(int, int) 是函数指针类型

void test(char* pc, int arr[10])
{
}
int main()
{
  void (*pf)(char *, int [10]) = test;
  return 0;
}

由上图可知:  

       函数名是函数的地址;

       &函数名也是函数的地址。

阅读两段有趣的代码:

//代码1

( * ( void ( * )()) 0 )();

解析:调用0地址处的函数

           1. 将0强制类型转换为void (*)()  类型的函数指针

           2. 调用0地址处的这个函数

//代码2

void ( * signal ( int , void ( * )( int )))( int );

解析:

   1.signal 是一个函数声明
   2.signal 函数有2个参数,第一个参数的类型是int,第二个参数的类型是 void(*)(int) 函数指针类型
   3.该函数指针指向的函数有一个int类型的参数,返回类型是void
   4.signal 函数的返回类型也是void(*)(int) 函数指针类型,该函数指针指向的函数有一个int类型的参数,返回类型是void

类型重定义:typedef

//类型重定义1
typedef unsigned int uint;
typedef int* ptr_t;
int main()
{
  uint u1;
  ptr_t p1;
  int* p2;
  return 0;
}
//类型重定义2
typedef int(*parr_t)[10];
typedef int (*pf_t)(int, int) ;
int main()
{
  typedef void(*pf_t)(int);
  pf_t signal(int, pf_t);
    //上方两句将下方的语句简化,效果相同
  void (* signal(int, void(*)(int) ) )(int);
  return 0;
}

函数指针数组:

       定义:int (*parr1[10])();  每个元素都是函数指针类型。

       用途:转移表。

函数指针数组的使用:

#include <stdio.h>
#include <string.h>
int Add(int x, int y)
{
  return x + y;
}
int Sub(int x, int y)
{
  return x - y;
}
int Mul(int x, int y)
{
  return x * y;
}
int Div(int x, int y)
{
  return x / y;
}
void menu()
{
  printf("***************************\n");
  printf("*****  1.add  2.sub  ******\n");
  printf("*****  3.mul  4.div  ******\n");
  printf("*****  0.exit        ******\n");
  printf("***************************\n");
}
//实现int类型的加减乘除
int main()
{
  int input = 0;
  int x = 0;
  int y = 0;
  int ret = 0;
  //函数指针数组的使用 - 转移表
  int (* pfArr[5])(int, int) = {NULL, Add, Sub, Mul, Div};
                                 0     1    2    3    4
  do
  {
    menu();
    printf("请选择:>");
    scanf("%d", &input);
    if (input >= 1 && input <= 4)
    {
      printf("请输入两个操作数:");
      scanf("%d %d", &x, &y);
      ret = pfArr[input](x, y);
      printf("ret = %d\n", ret);
    }
    else if(input == 0)
    {
      printf("退出计算器\n");
    }
    else
    {
      printf("选择错误,重新选择\n");
    }
  } while (input);
  return 0;
}

指向函数指针数组的指针:

定义:

       指向函数指针数组的指针是一个 指针 指针指向一个 数组 ,数组的元素都是 函数指针 ; (一般不直接写,通过函数指针一步一步变化得到,可以减少失误操作)

void (*pf)(const char*) = test;   //pf是函数指针变量

void (*pfArr[10])(const char*);  //pfArr是存放函数指针的数组

void (* (*p) [10])(const char*) = &pfArr;//p指向函数指针数组的指针

回调函数:

概念:

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

//回调函数的使用

void Calc(int (*pf)(int, int))

{

   int x = 0;

   int y = 0;

   int ret = 0;

   printf("请输入两个操作数:");

   scanf("%d %d", &x, &y);

  ret = pf(x, y);

   printf("ret = %d\n", ret);

}

使用回调函数模拟实现qsort()函数:

base:指向要排序的数组的第一个对象的指针,转换为 .void*。

num:数组中由指向的元素个数。是无符号整型。

size:数组中每个元素的大小(以字节为单位),是无符号整型。

compar:指向比较两个元素的函数的指针,重复调用此函数以比较两个元素。

qsort()运用:
#include <stdio.h>
//qosrt函数的使用者得实现一个比较函数
int int_cmp(const void * p1, const void * p2)
{
 return (*( int *)p1 - *(int *) p2);
}
int main()
{
 int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
 int i = 0;
 
 qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
 for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
 {
 printf( "%d ", arr[i]);
 }
 printf("\n");
 return 0;
}
排序int类型:
#include <stdio.h>
//比较int类型的比较函数
int my_compare(const void* q1, const void* q2)
{
  return (*(int*)q1 - *(int*)q2);
}
//交换每一个字节的元素
void Swap(char* b1, char* b2, int size)
{
  int i = 0;
  for (i = 0; i < size; i++)
  {
    char tmp = *b1;
    *b1 = *b2;
    *b2 = tmp;
    b1++;
    b2++;
  }
}
//模拟实现自己的qsort()函数
void my_qsort(void* base, int num, int size, int (*my_compare)(const void* q1, const void* q2))
{
  int i = 0;
  int j = 0;
  for (i = 0; i < num - 1; i++)
  {
    for (j = 0; j < num - 1 - i; j++)
    {
      //从小到大排序
      if (my_compare((char*)base+j*size,(char*)base+(j+1)*size) > 0)
      {
        Swap((char*)base + j*size, (char*)base + (j + 1)*size, size);
      }
    }
  }
}
int main()
{
  int arr[10] = { 2,4,6,7,8,3,1,0,9,5 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  my_qsort(arr, sz, sizeof(arr[0]), my_compare);
  return 0;
}
排序结构体类型:
#include <string.h>
//创建学生结构体
struct Stu
{
  char name[20];
  int age;
};
//比较int类型的比较函数
int my_compare_age(const void* q1, const void* q2)
{
  return ((struct Stu*)q1)->age - ((struct Stu*)q2)->age;
}
//比较int类型的比较函数
int my_compare_name(const void* q1, const void* q2)
{
  return strcmp( ( (struct Stu*)q1 )->name ,( (struct Stu*)q2 )->name);
}
//交换每一个字节的元素
void Swap(char* b1, char* b2, int size)
{
  int i = 0;
  for (i = 0; i < size; i++)
  {
    char tmp = *b1;
    *b1 = *b2;
    *b2 = tmp;
    b1++;
    b2++;
  }
}
//模拟实现自己的qsort()函数
void my_qsort(void* base, int num, int size, int (*my_compare)(const void* q1, const void* q2))
{
  int i = 0;
  int j = 0;
  //趟数
  for (i = 0; i < num - 1; i++)
  {
    //一趟内部比较的对数
    for (j = 0; j < num - 1 - i; j++)
    {
      //从小到大排序
      if (my_compare((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
      {
        //交换
        Swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
      }
    }
  }
}
int main()
{
  struct Stu arr[] = { {"zhangsan",34},{"lisi",27},{"wanwu",20} };
  int sz = sizeof(arr) / sizeof(arr[0]);
  my_qsort(arr, sz, sizeof(arr[0]), my_compare_age);
  my_qsort(arr, sz, sizeof(arr[0]), my_compare_name);
  return 0;
}

以上就是个人学习指针的个人见解和学习的解析,欢迎各位大佬在评论区探讨!

感谢大佬们的一键三连! 感谢大佬们的一键三连! 感谢大佬们的一键三连!

                                             

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