- 什么是函数指针
可以定义一个指向函数的指针变量,用来存放某一函数的起始地址,这就意味着此指针变量指向该函数。例如:
int (*p)(int,int);
定义p是指向函数的指针变量,它可以指向类型为整型且有两个整型参数的函数。p的类型用int (*)(int,int)表示
- 怎样定义和使用指向函数的指针变量
数据类型 (*指针变量名)(函数参数表列);
如 int (*p)(int,int);
p=max; 对
p=max(a,b); 错
p+n,p++,p--等运算无意义
例: 输入两个数,根据输入的字符
+,-,x,/算出对应的结果
解题思路:
1.定义两个float单精度浮点数变量a,b和一个字符变量c接收键盘的输入值及运算符号字符,
2.再定义一个函数指针和4个加,减,乘,除函数,
3.通过函数指针调用相应的函数进行运算
#include //包含标准输入输出头文件
// 加法
float add(float x,float y)
{
return x+y;
}
// 减法
float minus(float x,float y)
{
return x-y;
}
// 乘法
float mul(float x,float y)
{
return x*y;
}
// 除法
float div(float x,float y)
{
if(y != 0)
return x/y;
printf("除数不能为0\n");
return 0;
}
int main(void) //主函数,程序的入口
{
float a,b;
char c;
float (*func)(float,float); //函数指针
func = NULL;
printf("请输入2个运算数字,和1个加减乘除运算符\n");
scanf("%f,%f,%c",&a,&b,&c);
switch(c)
{
case '+':
func = add;
break;
case '-':
func = minus;
break;
case '*':
func = mul;
break;
case '/':
func = div;
break;
default:
printf("输入的运算符有误\n");
break;
}
if(func != NULL)
{
float result = func(a,b);
printf("result=%.2f\n",result);
}
return 0;
}
- 函数指针数组
#include //包含标准输入输出头文件
// 加法
float add(float x,float y)
{
return x+y;
}
// 减法
float minus(float x,float y)
{
return x-y;
}
// 乘法
float mul(float x,float y)
{
return x*y;
}
// 除法
float div(float x,float y)
{
if(y != 0)
return x/y;
printf("除数不能为0\n");
return 0;
}
int main(void) //主函数,程序的入口
{
float a,b;
char c;
int index;
float (*func[4])(float,float) = { // 函数指针数组
add,minus,mul,div
};
printf("请输入2个运算数字,和1个加减乘除运算符\n");
scanf("%f,%f,%c",&a,&b,&c);
switch(c)
{
case '+':
index = 0;
break;
case '-':
index = 1;
break;
case '*':
index = 2;
break;
case '/':
index = 3;
break;
default:
index = 4;
printf("输入的运算符有误\n");
break;
}
if(index < 4)
{
float result = func[index](a,b);
printf("result=%.2f\n",result);
}
return 0;
}
- 空类型指针(void*)
C语言中,void为“不确定类型”,不可以用void来声明变量。如:void a = 10;如果出现这样语句编译器会报错:variable or field ‘a’ declared void。
C语言中void * 为 “不确定类型指针”,void *可以用来声明指针。如:void * a;
(1)void *可以接受任何类型的赋值:
void *a = NULL;
int * b = NULL;
a = b;//a是void * 型指针,任何类型的指针都可以直接赋值给它,无需进行强制类型转换
(2)void *可以赋值给任何类型的变量 但是需要进行强制转换:
int * a = NULL ;
void * b ;
a = (int *)b;
void* 在转换为其他数据类型时,赋值给void* 的类型 和目标类型必须保持一致。简单点来说:void* 类型接受了int * 的赋值后 这个void * 不能转化为其他类型,必须转换为int *类型;
#include
int main ()
{
short a= 10;
void *b = &a;
printf(“short a = %d\n",a);
printf("void (short*)b =%d \n",*(short*)b);
printf(“void (double *)b =%d \n”,*(double*)b); //编译器并不会报错但是其结果却有点出人意料
return 0;
}
- 用指向函数的指针作函数参数
指向函数的指针变量的一个重要用途是把函数的地址作为参数传递到其他函数
指向函数的指针可以作为函数参数,把函数的入口地址传递给形参,这样就能够在被调用的函数中使用实参函数
- 返回指针值的函数
一个函数可以返回一个整型值、字符值、实型值等,也可以返回指针型的数据,即地址。其概念与以前类似,只是返回的值的类型是指针类型而已
定义返回指针值的函数的一般形式为
类型名 *函数名(参数表列);
- 什么是内存的动态分配
非静态的局部变量是分配在内存中的动态存储区的,这个存储区是一个称为栈的区域
C语言还允许建立内存动态分配区域,以存放一些临时用的数据,这些数据需要时随时开辟,不需要时随时释放。这些数据是临时存放在一个特别的自由存储区,称为堆区
#include
int a = 0; 全局初始化区
char *p1; 全局未初始化区
int main(void)
{
int b; 栈
char s[] = "abc"; 栈
char *p2; 栈
char *p3 = "123456"; 123456\0在常量区,p3在栈上。
static int c =0; 全局(静态)初始化区
p1 = (char *)malloc(10); 堆
p2 = (char *)malloc(20); 堆
return 0;
}
- 怎样建立内存的动态分配
对内存的动态分配是通过系统提供的库函数来实现的,主要有malloc,calloc,free,realloc这4个函数。
1.malloc函数
其函数原型为
void *malloc(unsigned int size);
其作用是在内存的动态存储区中分配一个长度为size的连续空间
函数的值是所分配区域的第一个字节的地址,或者说,此函数是一个指针型函数,返回的指针指向该分配域的开头位置
malloc(100);
开辟100字节的临时分配域,函数值为其第1个字节的地址
注意指针的基类型为void,即不指向任何类型的数据,只提供一个地址
如果此函数未能成功地执行(例如内存空间不足),则返回空指针(NULL)
2.calloc函数
其函数原型为
void *calloc(unsigned n,unsigned size);
其作用是在内存的动态存储区中分配n个长度为size的连续空间,这个空间一般比较大,足以保存一个数组。
用calloc函数可以为一维数组开辟动态存储空间,n为数组元素个数,每个元素长度为size。这就是动态数组。函数返回指向所分配域的起始位置的指针;如果分配不成功,返回NULL。如:
p=calloc(50,4);
开辟50×4个字节的临时分配域,把起始地址赋给指针变量p
3.free函数
其函数原型为
void free(void *p);
其作用是释放指针变量p所指向的动态空间,使这部分空间能重新被其他变量使用。p应是最近一次调用calloc或malloc函数时得到的函数返回值。
free(p);
释放指针变量p所指向的已分配的动态空间
free函数无返回值
4. realloc函数
其函数原型为
void *realloc(void *p,unsigned int size);
如果已经通过malloc函数或calloc函数获得了动态空间,想改变其大小,可以用recalloc函数重新分配。
用realloc函数将p所指向的动态空间的大小改变为size。p的值不变。如果重分配不成功,返回NULL。如
realloc(p,50);
将p所指向的已分配的动态空间改为50字节
以上4个函数的声明在stdlib.h头文件中,在用到这些函数时应当用“#include ”指令把stdlib.h头文件包含到程序文件中。
- 内存申请失败原因分析
假如有4k的地址空间,第一次分配了1k,第2次分配了2k,第3次分配了1k,刚好4k用完,然而程序释放了第1次的1k和第3次
的1k,内存有2k的空间可用,但这时要申请一块2k的内存,申请没有成功,原因是内存块虽然有2k的可用,但不是连续的2k,所
以导致申请内存失败
- 回调函数
1 定义和使用场合
回调函数是指使用者自己定义一个函数,实现这个函数的程序内容,然后把这个函数(入口地址)作为参数传入别人(或系统)的函数中,由别人(或系统)的函数在运行时来调用的函数。函数是你实现的,但由别人(或系统)的函数在运行时通过参数传递的方式调用,这就是所谓的回调函数。简单来说,就是由别人的函数运行期间来回调你实现的函数。
这一设计允许了底层代码调用在高层定义的子程序(如图1-1所示)。C语言中回调函数主要通过函数指针的方式实现。
解题思路:
- 定义一个1秒的定时器
- 定义一个钩子,捕捉按键消息
- 定义函数指针实现回调
- 定义一个注册函数,实现回调函数注册
- 实现消息循环,消息翻译,消息派遣
- 主函数中调用注册函数,实现自己编写的函数每2秒被回调
#include
#include
#include //windows窗口头文件
#define WH_KEYBOARD_LL 13
static int randomBuf[5]; //用于存放随机产生的数据
void (*customeRandom)(int *buf); //定义函数指针变量customeRandom
void (*resume)(int *buf); //定义函数指针变量resume
//控制台消息响应
void CALLBACK TimerProc(HWND hwnd, UINT uMsg, UINT idEvent, DWORD dwTime)
{
static int i;
if(++i % 2 == 0) //每2秒产生一组共5个随机数
{
if(customeRandom != NULL)
{
for(int j=0;j
关注
打赏
最近更新
- 深拷贝和浅拷贝的区别(重点)
- 【Vue】走进Vue框架世界
- 【云服务器】项目部署—搭建网站—vue电商后台管理系统
- 【React介绍】 一文带你深入React
- 【React】React组件实例的三大属性之state,props,refs(你学废了吗)
- 【脚手架VueCLI】从零开始,创建一个VUE项目
- 【React】深入理解React组件生命周期----图文详解(含代码)
- 【React】DOM的Diffing算法是什么?以及DOM中key的作用----经典面试题
- 【React】1_使用React脚手架创建项目步骤--------详解(含项目结构说明)
- 【React】2_如何使用react脚手架写一个简单的页面?