指针 --- 进阶

简介: 指针 --- 进阶

什么是指针,我们在之前的《指针》章节已经接触过了,我们知道了指针的概念:


1.指针就是个变量,用来存放地址,地址唯一标识一块内存空间。


2指针的大小是固定的4/8个字节( 32位平台/64位平台)。


3.指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。


4.指针的运算。可以比较大小,如果两个指针指向同一数组,也可以相减,得到数的绝对值是之间的元素个数。


1.字符指针

在指针类型中,我们知道有一种指针类型为字符指针 char* 

一般使用情况:

int main()
{
  char ch = 'w';
  char* pc = &ch;//pc就是字符指针,指向字符的指针就是字符指针
    *pc = 'w';
  return 0;
}


还有一种使用情况如下:存放常量字符串的首字符的地址

#include<stdio.h>
int main()
{
  //下面右边是一个表达式,表达式的值是首字符的地址
  char* p = "abcdef";//是把首字符的地址放到p中  
  //*p = 'w';//错误,这里字符串是常量字符串,不能修改  
  //const char *p = "abcdef";//这样写会更加严谨,
  char arr[] = "abcdef"; //可以想象为字符串是arr,p是存放的arr的首元素地址
  char* parr = arr;     // p指向的是arr首元素地址,不同的是arr数组是可以修改
  *parr='w';//可以修改arr字符串的值
  printf("%s\n", arr);
  return 0;
}


代码 char* p = "abcdef"; 特别容易让我们以为是把字符串 abcdef 放到字符指针 p 里了,但是/本质是把常量字符串 abcedf 首字符的地址放到了p中,也就是a的地址放到了指针变量p中。

看一下面这道题

#include <stdio.h>
int main()
{
  char str1[] = "hello xilanhua";
  char str2[] = "hello xilanhua";
  const char* str3 = "hello xilanhua";
  const char* str4 = "hello xilanhua";
  if (str1 == str2)
  {
    printf("str1 和 str2 相等\n");
  }
  else
  {
    printf("str1 和 str2 不相等\n");
  }
  if (str3 == str4)
  {
    printf("str3 和 str4 相等\n");
  }
  else
  {
    printf("str3 和 str4 不相等\n");
  }
  return 0;
}

str1 和str2 是两个字符数组的首元素地址,在内存中开辟不同的空间,所以str1和str2不相等,


str3 , str4 存放常量字符串 "hello xilanhua" 的首字符地址,常量字符串不能被修改,所以在内存中没有必要存两份,只需要开辟一份空间,两个指针都指向的这份空间,所以两个地址相同的。


所以输出结果为

2.指针数组

可以通过类比:

整型数组,存放整形的数组

字符数组,存放字符的数组

指针数组,存放指针的数组。

int* arr1[5];//整形指针数组
char* arr2[5];//一级字符指针的数组
char** arr3[5];//二级字符指针的数组

根据上面学的字符指针,我们看一下下面代码:

#include <stdio.h>
int main()
{
  char* arr[] = { "abcdef","hehe","xilanhua" };//字符指针数组
  //               arr0    arr1   arr2    //这里存放的是常量字符串的首字符的地址
  //*arr[1] = 'w';//错误,这里数组元素指向的也是常量字符串,不能修改
  for (int i = 0; i < 3; i++)
  {
    printf("%s\n", arr[i]);
  }
  return 0;
}


结果

看这段代码:


#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 };
  //数组名是首元素地址
  //arr是存放整形指针的数组    指针数组
  int* arr[] = { arr1,arr2,arr3 };
  int i = 0;
  int j = 0;
  for (i = 0; i < 3; i++)
  {
    for (j = 0; j < 5; j++)
    {
      printf("%d ", arr[i][j]);//和二维数组很像
      //printf("%d ", *(arr[i] + j));
            //*(arr[i]+j)==arr[i][j]     arr[i]==*(arr+i)
    }
    printf("\n");
  }
  return 0;
}


结果:

可以发现和二维数组很像,但是本质上是不同的,二维数组是连续存放的空间,但是这里不是,是通过指针联系起来的。

3.数组指针

也是通过类比:

整形指针,指向整形的指针变量

字符指针,指向字符的指针变量

数组指针,指向数组的指针变量


下面哪一个是数组指针?

int* p1[10];

int (*p2)[10];

解释:

int (*p)[10]

p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
这里要注意:[ ] 的优先级要高于*号的,所以必须加上()来保证p先和*结合。不加(),就是指针数组。


1. &数组名 和 数组名

看下面代码:

#include<stdio.h>
int main()
{
  int arr[10] = { 0 };
  printf("%p\n", arr);//数组名是首元素地址 两个例外 &数组名和sizeof(数组名)
  printf("%p\n", &arr[0]);//首元素地址
  printf("%p\n", &arr);  //三个数值上相等
  printf("%p\n", arr + 1);//int* + 1
  printf("%p\n", &arr[0] + 1);//int* + 1
  printf("%p\n", &arr + 1);//加了40 int(*)[10]+1
  int(*p)[10] = &arr;
  //int(*)[10]  这个数组指针的类型
  return 0;
}


根据上面的代码我们发现,其实&ar r和 arr,虽然值上是相同的,但是意义是不相同的。实际上:&arr表示的是数组的地址,而不是数组首元素的地址。(细细体会一下)。数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于&arr 的差值是40.


总结:


数组名是首元素地址 有两个例外 &数组名 和 sizeof(数组名)

1.sizeof(arr)  -  sizeof内部单独放一个数组名的时候,数组名表示整个数组,计算得到的是数组的大小

2.&arr   -   这里的数组名表示的是整个数组,取出的是整个数组的地址,从地址值的角度来讲和数组的首元素地址是一样的,但是意义不一样  


2.访问数组元素的不同方式

#include<stdio.h>
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int i = 0;
  int sz = sizeof(arr) / sizeof(arr[0]);
  //下标形式访问数组
  for (i = 0; i < sz; i++)
  {
    printf("%d ", arr[i]);
  }
  //使用指针访问数组
  int* p = arr;
  for (i = 0; i < sz; i++)
  {
    for (i = 0; i < sz; i++)
    {
      printf("%d ", *(p + i)); 
    }
  }
  //数组指针访问数组   虽然对,但是不推荐
  int(*pa)[10] = &arr; 
  //pa == &arr
  //*pa == *&arr
  //*pa == arr 数组名 不是两种特殊情况 是首元素地址
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *((*pa)+i));//(*pa)[i];
  }
  return 0;
}


注意:

int(*pa)[10] = &arr;
   //pa == &arr
   //*pa == *&arr
   //*pa == arr 数组名 不是两种特殊情况 是首元素地址

所以:数组指针 解引用 是数组首元素地址


3. 数组指针的使用

二维数组的传参要使用到指针数组。

#include<stdio.h>
//普通接收二维数组传参形式
void print1(int arr[3][5], int r, int c)
{
  int i = 0;
  int j = 0;
  for (i = 0; i < r; i++)
  {
    for (j = 0; j < c; j++)
    {
      printf("%d ", arr[i][j]);
    }
    printf("\n");
  }
}
//二维数组传参用到数组指针
void print(int (*arr)[5], int r, int c)//指针形式接受  数组指针
{
  int i = 0;
  int j = 0;
  for (i = 0; i < r; i++)//arr+i是每一行  *(arr+i)是每一行的数组名 arr[i] 首元素地址就是每一行的第一个元素的地址
  {
    for (j = 0; j < c; j++)
    {
      //printf("%d ", *(*(arr + i) + j));//解引用数组指针就是数组首元素地址,二维数组是第一行地址
      printf("%d ", arr[i][j]);//*(*(arr+i)+j)==*(arr+i)[j]==arr[i][j]
    }
    printf("\n");
  }
}
int main()
{
  int arr[3][5] = { 1,2,3,4,5,2,3,4,5,6,3,4,5,6,7 };
  print(arr, 3, 5);//二维数组的数组名,也表示首元素地址
  //二维数组的首元素地址是第一行的地址,存放的是一维数组   
  //可以理解为二维数组是存放一维数组的数组
  return 0;
}

二维数组的数组名,也表示首元素地址。二维数组的首元素地址是第一行的地址,存放的是一维数组。可以理解为二维数组是存放一维数组的数组

4.数组传参和指针传参


在写代码的时候难免要把【数组】或者【指针】传递给函数,那函数的参数该如何设计呢?

一维数组传参:

形参可以是数组,也可以是指针,当形参是指针时,要注意类型。

#include<stdio.h>
void test(int arr[])//不会真的创建一个数组,不写大小
{}
void test(int arr[10])//和传入格式一样,不会真的创建
{}
void test(int* arr)//数组名,首元素地址,指针接受
{}
void test2(int* arr[20])//指针数组,和传入格式一样
{}
void test2(int** arr )//数组首元素是int* 类型,*arr说明他是指针
{}
int main()
{
  int arr[10] = { 0 };//整形数组
  int* arr2[20] = { 0 };//整形指针数组
  test(arr);
  test2(arr2);
  return 0;
}


二维数组传参:

参数可以是指针,也可以是数组,如果是数组,行可以省略,列不能省略,如果是指针,传参传过去的是第一行的指针,形参就应该是数组指针

void test(int arr[3][5])
{}
//void test(int arr[][])//错误,只能省略行,因为得知道一行有几个元素
//{}
void test(int arr[][5])
{}
//void test(int *arr)//错误,二维数组的首元素地址是第一行的地址
//{}
//void test(int* arr[5])//错误,这是一个指针数组,存放整形指针的数组
//{}
void test(int(*arr)[5])//数组指针,可以接收实参
{}
//void test(int** arr)//错误,二级指针
//{}
int main()
{
  int arr[3][5] = { 0 };
  test(arr);
  return 0;
}

一级指针传参:

#include<stdio.h>
void print(int* p, int sz)
{
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d\n", *(p + i));//通过指针访问
  }
}
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  int* p = arr;//一级指针
  int sz = sizeof(arr) / sizeof(arr[0]);//元素个数
  print(p, sz);
  return 0;
}

思考:当一个函数参数是一个指针时,函数能接收什么参数


void print(int* p);    

1.int a;

 print(&a);//变量的地址

2.int* p=&a;

 print(p);//指针变量

3.int arr[10];

 print(arr);//数组名


二级指针传参:

#include<stdio.h>
//用二级指针接收
void test(int** ptr)
{
  printf("num = %d\n", **ptr);
}
int main()
{
  int n = 10;
  int* p = &n;
  int** pp = &p;
  test(pp);
  test(&p);
  return 0;
}

思考:当函数的参数为二级指针的时候,可以接收什么参数

1.一级指针的地址

2.二级指针变量,

3.指针数组



5.函数指针

这里还是类比:

整形指针,指向整形的指针 int*

字符指针,指向字符的指针 char*

数组指针,指向数组的指针,int arr[10];  int (*p)[10] = &arr;

函数指针,指向函数的指针, 即存放函数的地址


那么问题来了,函数有地址吗

#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
//&函数名得到就是函数的地址
int main()
{
  printf("%p\n", &Add);//可以运行
  printf("%p\n", Add);//可以运行
  return 0;
}
//函数名 和 &函数名都是函数的地址


结果:

可以得出结论,函数名 和 &函数名 都是函数的地址

使用函数指针调用函数:

#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
int main()
{
  //()是pf先于*结合,说明pf是指针
  int (*pf)(int, int) = Add;//函数的地址要存起来,就得放在 函数指针变量pf 中
  //最前面的是函数返回类型,后面括号内是函数参数类型
  //通过函数指针调用函数
  int ret = (*pf)(3, 5);
  ret = Add(3, 5);//函数可以这样调用,Add是地址
  ret = pf(3, 5);//所以这样写也对
  ret = (*****pf)(3, 5);//编译器在处理时,会把*去掉,也没有问题
  printf("%d\n", ret);//8
  return 0;
}


可以阅读两段有趣的代码

1.(*(void(*)( ))0)( );

#include<stdio.h>
int main()
{
  //1.将0强制类型转换为void(*)()类型的函数指针
  //2.这就意味着0地址处放着一个函数,函数无参,返回类型是void
  //3.调用0地址处的这个函数
  (*(void(*)())0)();//前面*可以不写
  return 0;
}

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

int main()
{
  void(* signal(int, void(*)(int) ) )(int);
  //signal先于()结合
  //void(*)(int)
  //signal(int, void(*)(int));//函数名称和函数参数类型
  //上述代码是一个函数的声明,
  //函数的名字是signal
  //函数的参数第一个是int类型,第二个是void(*)(int)类型的函数指针
  //该函数指针指向的函数参数是int,返回类型是void
  //signal函数的返回类型也是一个函数指针
  //该函数指针指向的函数参数是int,返回类型是void
  return 0;
}
可以通过类型重命名简化为:
//typedef int(*)(int) pt_f;//写法不对
typedef void(*pf_t)(int);//将void(*)(int)类型的函数指针重命名叫pf_t 
pf_t(signal(int, pf_t));

6.函数指针数组

存放函数指针的数组

函数指针数组写法

函数指针
int (*p)(int,int);
函数指针数组只需在p后加一个[大小],让指p先于[]结合
int (*p[10])(int,int);
//数组存放元素的类型就是 int (*) (int,int)

函数指针数组的用途:转移表

例子:(计算器)

#include<stdio.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");
}
int main()
{
  //转移表 - 函数指针数组   函数指针类型要相同
  int(*pfArr[5])(int, int) = { NULL,Add,Sub,Mul,Div };//NULL==0
  //            下标 0  1  2    3   4
  int input = 0;
  int x = 0;
  int y = 0;
  int ret = 0;
  do
  {
    menu();
    printf("请输入:> ");
    scanf("%d", &input);
    if (input == 0)
    {
      printf("退出计算器\n");
      break;
    }
    if (input <= 4 && input >= 1)
    {
      printf("请输入两个操作数:>");
      scanf("%d %d", &x, &y);
      printf("%d\n",pfArr[input](x, y));//调用相应的计算函数
    }
    else
    {
      printf("输入非法,请重新输入\n");
    }
  } while (input);
  return 0;
}

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

指向函数指针数组的指针是一个指针,指针指向一个数组,数组的元素都是函数指针

示例:

#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
int Sub(int x, int y)
{
  return x - y;
}
int main()
{
    //函数指针
  int(*pf)(int, int) = Add;
  //函数指针数组
  int(*pfArr[4])(int, int) = { Add,Sub };
  //ppfArr是一个指向一个函数指针数组的指针变量
  int(*(*ppfArr)[4])(int,int) = &pfArr;
  return 0;
}


8.回调函数

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


这里演示一个用到回调函数的库函数,qsort 函数,包含头文件 stdlib.h 。


我们通过c++官方网站旧版上可以查到 qsort 函数的用法qsort用法


先看返回值类型和参数列表


它可以排序任何类型的数据,但是需要自己写一个函数用来可以确定两个元素的大小,也就是参数列表中的compar函数。

void qsort( void* base,//待排序数组的第一个元素
      size_t num, //待排序的元素个数
      size_t size, //每个元素的大小
      int(*cmp)(const void*, const void*));//函数指针,指向一个函数,这个函数可以比较2个元素的大小

qsort 函数底层使用的是快速排序,也是一种排序算法,我们这里用冒泡排序实现一下。


因为这个函数可以排序任何类型,所以我们最开始用  void * 指针来接收传来的指针,然后再通过强制类型转换 (char*) 再加上 size 来确定操作空间的大小。


我们先来实现以下cmp 函数


这是cmp函数的要求,返回值有大于0,小于0,和等于0,三种情况。 大于0指向p1,p1大,小于0指向p2,p2大,相等返回值是0。

int cmp_int(const void* p1, const void* p2)
{
  return *(int*)p1 - *(int*)p2;
    //这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*)
}


交换函数:

因为我们不知道要交换什么类型元素,所以是使用 (char*) 强制类型转换,每次交换一个字节,循环size次的方法

void Swap(char* p1, char* p2, size_t size)
{
  for (size_t i = 0; i < size; i++)
  {
    char t = *(p1 + i);
    *(p1 + i) = *(p2 + i);
    *(p2 + i) = t;
  }
}


排序内部:

void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*))
{
  size_t i = 0;
  size_t j = 0;
  for (i = 0; i < num - 1; i++)//一趟冒泡排序
  {
    for (j = 0; j < num - 1 - i; j++)
    {
      //两个相邻元素比较
      //arr[j]与arr[j+1]  调用我们自己写的比较函数,回调函数
      if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
      {
        //交换
        Swap((char*)base + j * size, (char*)base + (j + 1) * size,size);
      }
    }
  }

全部代码:

#include<stdio.h>
#include<stdlib.h>
int cmp_int(const void* p1, const void* p2)
{
  return *(int*)p1 - *(int*)p2;
  //这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*)
}
void Swap(char* p1, char* p2, size_t size)
{
  for (size_t i = 0; i < size; i++)
  {
    char t = *(p1 + i);
    *(p1 + i) = *(p2 + i);
    *(p2 + i) = t;
  }
}
void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*))
{
  size_t i = 0;
  size_t j = 0;
  for (i = 0; i < num - 1; i++)//一趟冒泡排序
  {
    for (j = 0; j < num - 1 - i; j++)
    {
      //两个相邻元素比较
      //arr[j]与arr[j+1]
      if (cmp((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] = { 1,2,8,9,10,5,3,6,4,7 };
  qsort(arr, 10, sizeof(arr[0]), cmp_int);
  for(int i=0;i<10;i++)
  {
    printf("%d ", arr[i]);
  }
  printf("\n");
  int arr1[10] = { 1,2,8,9,10,5,3,6,4,7 };
  bubble_sort(arr1, 10, sizeof(arr[0]), cmp_int);
  for (int i = 0; i < 10; i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}


本篇结束

相关文章
|
5月前
|
C语言
指针进阶(C语言终)
指针进阶(C语言终)
|
5月前
|
机器学习/深度学习 搜索推荐 算法
【再识C进阶2(下)】详细介绍指针的进阶——利用冒泡排序算法模拟实现qsort函数,以及一下习题和指针笔试题
【再识C进阶2(下)】详细介绍指针的进阶——利用冒泡排序算法模拟实现qsort函数,以及一下习题和指针笔试题
|
5月前
|
C语言
指针进阶(回调函数)(C语言)
指针进阶(回调函数)(C语言)
|
5月前
|
存储 C语言 C++
指针进阶(函数指针)(C语言)
指针进阶(函数指针)(C语言)
|
5月前
|
编译器 C语言
指针进阶(数组指针 )(C语言)
指针进阶(数组指针 )(C语言)
|
5月前
|
搜索推荐
指针进阶(2)
指针进阶(2)
47 4
|
5月前
指针进阶(3)
指针进阶(3)
41 1
|
5月前
|
C++
指针进阶(1)
指针进阶(1)
43 1
|
5月前
|
存储 安全 编译器
C++进阶之路:何为引用、内联函数、auto与指针空值nullptr关键字
C++进阶之路:何为引用、内联函数、auto与指针空值nullptr关键字
42 2
|
5月前
|
Java 程序员 Linux
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
49 0