C语言-指针进阶(下)

简介: C语言-指针进阶(下)


六、函数指针数组



      函数指针数组:把函数的地址存到一个数组中。

int (*parr1[10])();


parr1 先和 [] 结合,parr1是数组,数组的内容是 int (*)() 类型的函数指针。函数指针数组的用途:转移表。

常规调用函数实现简易计算器

#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("****5 Exit        ****\n");
  printf("**********************\n");
}
int main()
{
  int x = 0, y = 0, input = 0,ret=0;
  do
  {
    menu();
    printf("请输入:");
    scanf("%d", &input);
    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 5:
        printf("退出计算器\n");
        break;
      default:
        printf("输入数字无效,请重新输入\n");
        break;
    }
  } while (input);
  return 0;
}


用函数指针实现简易计算器


#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("****5 Exit        ****\n");
  printf("**********************\n");
}
int main()
{ 
  int x = 0, y = 0, input = 0, ret = 0;
  do
  {
    menu();
    printf("请输入:");
    scanf("%d", &input);
    printf("请输入两个整数:");
    scanf("%d%d",& x, &y);
    int(*pf[5])(int, int) = { NULL,Add,Sub,Mul,Div };//转移表
    if (input >= 1 && input <= 4)
    {
      ret = pf[input](x, y);
      printf("%d\n", ret);
    }
    else if (input == 5)
      printf("退出计算机");
    else
      printf("输入数字无效,请重新输入:\n");
  } while (input);
  return 0;
}



七、指向函数指针数组的指针



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

void test ( const char* str )
{
   printf ( "%s\n" , str );
}
int main ()
{
   // 函数指针 pfun
   void ( * pfun )( const char* ) = test ;
   // 函数指针的数组 pfunArr
   void ( * pfunArr [ 5 ])( const char* str );
   pfunArr [ 0 ] = test ;
   // 指向函数指针数组 pfunArr 的指针 ppfunArr
   void ( * ( * ppfunArr )[ 5 ])( const char* ) = & pfunArr ;
   return 0 ;
}


八、回调函数



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


用回调函数实现简易计算器

#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("****5 Exit        ****\n");
  printf("**********************\n");
}
void Cala(int(*p)(int, int))//回调函数
{
  int x = 0, y = 0, ret = 0;
  printf("请输入两个数:");
  scanf("%d %d", &x, &y);
  ret = p(x, y);
  printf("%d\n", ret);
}
int main()
{
  int x = 0, y = 0, input = 0, ret = 0;
  do
  {
      menu();
      printf("请输入:");
      scanf("%d", &input);
      switch (input)
      {
        case 1:
            Cala(Add);
            break;
        case 2:
          Cala(Sub);
        case 3:
            Cala(Mul);
            break;
        case 4:
          Cala(Div);
            break;
        case 5:
            printf("退出计算器\n");
            break;
        default:
            printf("输入数字无效,请重新输入\n");
            break;
      }
  } while (input);
  return 0;
}



九、指针和数组笔试题



9.1 一维数组


int a [] = { 1 , 2 , 3 , 4 };
printf ( "%d\n" , sizeof ( a ));//整个数组的大小,16个字节
printf ( "%d\n" , sizeof ( a + 0 ));//数组名a是首元素地址,a+0还是首元素的地址,地址大小为4/8个字节
printf ( "%d\n" , sizeof ( * a ));//数组名a是首元素地址,*a就是首元素,大小就是4个字节
printf ( "%d\n" , sizeof ( a + 1 ));//数组名a是首元素地址,数组名a+1是第二个元素地址,是地址大小就是4/8个字节
printf ( "%d\n" , sizeof ( a [ 1 ]));//元素大小,4个字节
printf ( "%d\n" , sizeof ( & a ));//&a是数组的地址,是地址就是4/8个字节
printf ( "%d\n" , sizeof ( *& a ));// sizeof ( *& a )== sizeof ( a ),16个字节
printf ( "%d\n" , sizeof ( & a + 1 ));//&a是数组的地址,&a+1就是跳过此数组的地址,是地址就是4/8个字节
printf ( "%d\n" , sizeof ( & a [ 0 ]));//&a[0]首元素地址,4/8个字节
printf ( "%d\n" , sizeof ( & a [ 0 ] + 1 ));第二个元素的地址,4/8个字节


9.2 字符数组


char arr [] = { 'a' , 'b' , 'c' , 'd' , 'e' , 'f' };
printf ( "%d\n" , sizeof ( arr ));// 整个数组大小,6个字节
printf ( "%d\n" , sizeof ( arr + 0 ));//数组名arr是首元素地址,a+0还是首元素的地址,地址大小为4/8个字节
printf ( "%d\n" , sizeof ( * arr ));//数组名arr是首元素地址,*arr就是首元素,大小就是1个字节
printf ( "%d\n" , sizeof ( arr [ 1 ]));//元素大小,1个字节
printf ( "%d\n" , sizeof ( & arr ));//&arr是数组的地址,是地址就是4/8个字节
printf ( "%d\n" , sizeof ( & arr + 1 ));//&arr是数组的地址,&arr+1就是跳过此数组的地址,是地址就是4/8个字节
printf ( "%d\n" , sizeof ( & arr [ 0 ] + 1 ));//&a[0]首元素地址,&a[0]+1是第二个元素的地址,4/8个字节
printf ( "%d\n" , strlen ( arr ));//因为字符数组arr中没有\0,所以在求字符串长度的时候,会一直往后找,产生的结构就是随机值
printf ( "%d\n" , strlen ( arr + 0 ));//arr + 0是首元素的地址,和第一个一样,也是随机值
printf ( "%d\n" , strlen ( * arr )); //error,strlen函数参数的部分需要传一个地址,当我们传递是'a'时,'a'的ASCII码值是97,那就是将97作为地址传参,strlen就会从97这个地址开始统计字符串长度,这就非法访问内存了
printf ( "%d\n" , strlen ( arr [ 1 ]));//如上同样问题
printf ( "%d\n" , strlen ( & arr ));//&arr是数组的地址,数组的地址和数组首元素的地址,值是一样的,那么传递给strlen函数后,依然是从数组的第一个元素的位置开始往后统计
printf ( "%d\n" , strlen ( & arr + 1 ));//随机值
printf ( "%d\n" , strlen ( & arr [ 0 ] + 1 ));&arr[0] + 1是第二个元素的地址。结果也是随机值


char arr [] = "abcdef" ;//[a b c d e f \0]
printf ( "%d\n" , sizeof ( arr ));//数组大小,7
printf ( "%d\n" , sizeof ( arr + 0 ));//首元素地址大小,4/8
printf ( "%d\n" , sizeof ( * arr ));//首元素大小,1
printf ( "%d\n" , sizeof ( arr [ 1 ]));//第二个元素大小。1
printf ( "%d\n" , sizeof ( & arr ));//地址,4/8
printf ( "%d\n" , sizeof ( & arr + 1 ));//地址
printf ( "%d\n" , sizeof ( & arr [ 0 ] + 1 ));//地址
printf ( "%d\n" , strlen ( arr ));//6
printf ( "%d\n" , strlen ( arr + 0 ));//6
printf ( "%d\n" , strlen ( * arr ));//error
printf ( "%d\n" , strlen ( arr [ 1 ]));//error
printf ( "%d\n" , strlen ( & arr ));//首地址,6
printf ( "%d\n" , strlen ( & arr + 1 ));//跳过整个数组的地址,随机值
printf ( "%d\n" , strlen ( & arr[0]+1);//从第二个元素的地址开始查找,5


char * p = "abcdef" ;//p指针变量,存放地址
printf ( "%d\n" , sizeof ( p ));//  大小是4/8
printf ( "%d\n" , sizeof ( p + 1 ));//p+1是b的地址
printf ( "%d\n" , sizeof ( * p ));//*p是首元素  1
printf ( "%d\n" , sizeof ( p [ 0 ]));//首元素   
printf ( "%d\n" , sizeof ( & p )); //地址
printf ( "%d\n" , sizeof ( & p + 1 ));//跳过p所在地址,还是地址
printf ( "%d\n" , sizeof ( &p[0] + 1 ));//b的地址
printf ( "%d\n" , strlen ( p ));//6
printf ( "%d\n" , strlen ( p + 1 ));//p+1-->b,5
printf ( "%d\n" , strlen ( * p ));//error
printf ( "%d\n" , strlen ( p [ 0 ]));//error
printf ( "%d\n" , strlen ( & p ));//随机值
printf ( "%d\n" , strlen ( & p + 1 ));//随机值
printf ( "%d\n" , strlen ( & p [ 0 ] + 1 ));//p+1-->b,5


9.3 二维数组


int a [ 3 ][ 4 ] = { 0 };
printf ( "%d\n" , sizeof ( a ));//整个数组大小 48个字节
printf ( "%d\n" , sizeof ( a [ 0 ][ 0 ]));//第1行第1列元素 ,4个字节
printf ( "%d\n" , sizeof ( a [ 0 ]));//第1行的大小 ,16个字节
printf ( "%d\n" , sizeof ( a [ 0 ] + 1 ));//a[0]可以是第1行的数组名,在这里a[0]是是数组首元素地址,也就是a[0][0]的地址,其加1为第1行第1列元素地址,大小为4/8个字节
printf ( "%d\n" , sizeof ( * ( a [ 0 ] + 1 )));//第2行大小,4个字节
printf ( "%d\n" , sizeof ( a + 1 ));//a是数组首元素地址就是第一行地址,a+1就是第二行地址,4/8个字节
printf ( "%d\n" , sizeof ( * ( a + 1 )));//第二行数组大小,16个字节
printf ( "%d\n" , sizeof ( & a [ 0 ] + 1 ));//第二行地址大小,4/8个字节
printf ( "%d\n" , sizeof ( * ( & a [ 0 ] + 1 )));//第二行数组大小,16字节
printf ( "%d\n" , sizeof ( * a ));//a是第一行地址,大小16字节
printf ( "%d\n" , sizeof ( a [ 3 ]));//a[3]-->int [4]   16字节


总结:

       1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。

       2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。

       3. 除此之外所有的数组名都表示首元素的地址。

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