C语言进阶之指针(2)

简介: C语言进阶之指针(2)

前情回顾

指针数组和数组指针

1|int arr[5];
2|int* parr1[10];
3|int (*parr2)[10];
4|int (*parr3[10])[5];

1|arr是整形数组

2|parr1是整形指针数组

3|parr2是数组指针

4|parr3与[10]结合,是一个数组,数组的类型是int (*) [5],即数组指针类型,int (*parr3[10])[5]是数组指针数组,是用来存放数组指针的数组。

数组指针数组parr中,存放着10个数组指针,每个数组指针指向一个数组。

1.数组参数,指针参数

当我们要把【数组】或者【指针】作为参数传给函数,函数的参数应该如何设计

1.1一维数组传参

#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 main()
{
    int arr[10] = {0};
    int* arr2[20] = {0};
    test(arr);
    test2(arr2);
}

以上的写法都是正确的,一维数组的传参可以用数组的形式接收,也可以用指针的形式去接受,

test(arr),传过去的是数组arr首元素的地址,test(arr2),传过去的是数组arr2首元素的地址。

test(arr)

1|可以通过数组来接收,int arr[10],int arr[],数组的元素的个数可以省略

2|可以通过指针来接收,int* arr ,传来的是int 类型的地址

tes2(arr2)

1|可以通过数组来接收,int* arr[20],数组arr2的元素都是int*类型的,因此也应该用int*类型的数组来接收

2|可以通过指针来接收,int** arr,数组arr2的元素都是int*类型的,arr2表示的是首元素的地址,作为一个一级指针的地址,就要用二级指针来接收。

1.2二维数组传参

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);
}

二维数组传参,二维数组名arr表示的是首元素的地址,二维数组首元素的地址表示的是第一行的地址,即为一个一维数组的地址。

1|通过数组来接收,int arr[3][5],int arr[][5]二维数组的行可以省略,但是列不能省略。

2|通过指针来接收,传过去的arr为一维数组的地址,就要通过数组指针来接收,int (*arr)[5].

3|int* arr[5]是指针数组,表示的是数组,不能够接收。

1.3一级指针传参

#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};
    int* p = arr;
    int sz = sizeof(arr)/sizeof(arr[0]);
    //一级指针怕,传给函数
    print(p,sz);
    return 0;
}

函数的参数部分是指针 ,一级指针传过去,一级指针来接收。

传过去的实参和形参能够匹配上就是合适的。

1.4二级指针传参

#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;
}

二级指针传参,实参可以是二级指针pp,也可以是一级指针的地址&p

二级指针其实就表示一级指针的地址。

思考:

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

void test(char** p)
{}
int main()
{
    char c='b';
    char* pc =&C;
    char** ppc = &pc;
    char* arr[10];
    test(&pc);
    test(ppc);
    test(arr);
    return 0;
}

函数test 的参数是char** p,char类型的二级指针。

1|一级指针的地址,&p

2|二级指针, ppc

3|char*类型的数组名,数组名arr表示首元素的地址,即为一级指针的地址

2.函数指针

2.1函数指针的定义

函数指针--指向函数的指针

int Add(int x,int y)
{
     return x+y;
}
int main()
{
    int arr[5]={ 0 };
    //&数组名 - 取出的数组的地址
    int(*p)[5] = &arr;//数组指针
     //&函数名——取出的是函数的地址
    printf("%p\n",&Add);   
    printf("%p\n",Add);
    //函数指针创立的形式
    int (*pf)(int ,int) =&Add;
    //int (*pf)(int ,int) =Add;
}

对函数来说&函数名和函数名都是函数的地址。

int (*p)(int x,int y),可以省略x,y,主要把参数的类型表达清楚。

我们可以取Add的地址传给pf,也可以直接将Add直接传给pf

2.2函数指针的使用

int (*p)(int ,int) =&Add;
int ret = (*pf)(2,3);
printf("%d\n",ret);//5
//int ret = pf(2,3)
//指针pf的*可以去掉

int (*pf)(int ,int) =Add,这里Add直接传给pf,pf也可以直接表示Add。

int Add(int x,int y)
{
    return x+y;
}
void calc(int(*pf)(int,int))
{
    int a = 3;
    int b = 5;
    int ret = pf(a,b);
    printf("%d\n",ret);
}
int main()
{
    calc(Add);
    return 0;
}

通过函数Add传参,可以使另一个函数calc使用函数Add .

3.函数指针数组

函数指针数组是一个数组,用来存放函数指针。

3.1函数指针的用途

计算器函数的实现

void menu()
{
  printf("******************************\n");
  printf("*****   1.add   2.sub    *****\n");
  printf("*****   3.mul   4.div    *****\n");
  printf("*****   0.exit           *****\n");
  printf("******************************\n");
}
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;
}
int main()
{
  int input = 0;
  int x = 0;
  int y = 0;
  do
  {
    menu();
    printf("请选择》:");
    scanf("%d", &input);
    int ret = 0;
    switch (input)
    {
    case 1:
      printf("请输入两个操作数》:");
      scanf("%d%d", &x, &y);
      ret = Add(x, y);
      printf("%d\n", ret);
      break;
    case 2:
      printf("请输入两个操作数》:");
      scanf("%d%d", &x, &y);
      ret = Sub(x, y);
      printf("%d\n", ret);
      break;
    case 3:
      printf("请输入两个操作数》:");
      scanf("%d%d", &x, &y);
      ret = Mul(x, y);
      printf("%d\n", ret);
      break;
    case 4:
      printf("请输入两个操作数》:");
      scanf("%d%d", &x, &y);
      ret = Div(x, y);
      printf("%d\n", ret);
      break;
    case 0:
      printf("退出计算器\n");
      break;
    default:
      printf("选择错误\n");
      break;
    }
  } while (input);
  return 0;
}
printf("请输入两个操作数》:");
scanf("%d%d", &x, &y);
ret = Add(x, y);
printf("%d\n", ret);

其中会有很多冗余的代码,这里就可以使用函数指针来解决。

上面的这段可以改为calc(Add);,然后在创立一个calc的函数。

case 1,case 2,case 3,case 4大部分内容都是相同的,不同的就是运用的函数不一样(Add,Sub,Mul,Div)

void calc(int (*pf)(int ,int ))
{
    int x = 0;
    int y = 0;
    int ret = 0;
    printf("请输入2个操作数:>")
    scanf("%d%d",&x,&y);
    ret = pf(x,y);
    printf("%d\n",ret);
}

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

存放函数指针的数组就是函数指针数组

那么函数指针的表现形式应该是怎样的呢?

int main()
{
    int (*pf)(int,int) = Add;//pf是函数指针
    //函数指针数组用来存放函数指针
    //函数指针数组中的每个元素的类型是int (*)(int,int)
    int (*arr[4])(int,int)={ Add,Sub,Mul,Div};
    return 0;
}

通过函数指针数组arr,可以找到函数(Add,Sub,Mul,Div)。

通过函数指针数组,我们的计算器程序就可以进行进一步的优化

void menu()
{
  printf("******************************\n");
  printf("*****   1.add   2.sub    *****\n");
  printf("*****   3.mul   4.div    *****\n");
  printf("*****   0.exit           *****\n");
  printf("******************************\n");
}
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;
}

上面的这些内容还是不变,

int main()
{
    int input = 0;
    int x = 0;
    int y = 0;
    int ret = 0;
    int (*pfArr[5])(int,int) = { 0,Add,Sub,Mul,Div};
    do
    {
        menu();
        printf("请选择》:");
        scanf("%d",&input);
        if(input == 0)
        {
            printf("退出计算器\n");
        }
        else if(input >=1 && input<=4)
        {
            printf("请输入2个操作数:》");
            scanf("%d%d",&x,&y);
            ret =pfArr[input](x,y);
            printf("%d\n",ret);
        }
        else
        {
            printf("输入错误\n");
        }
    }while(input);
    return 0;
}

函数指针数组可以方便以后添加新的功能,通过输入下标的方式,就可以直接使用我们的函数。

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