符合中小企业对网站设计、功能常规化式的企业展示型网站建设
本套餐主要针对企业品牌型网站、中高端设计、前端互动体验...
商城网站建设因基本功能的需求不同费用上面也有很大的差别...
手机微信网站开发、微信官网、微信商城网站...
学习了数组之后,我们知道数组是在内存中申请一块内存空间;数组名代表内存块的首地址,通过数组名可以访问内存块中的数据。
站在用户的角度思考问题,与客户深入沟通,找到桥西网站设计与桥西网站推广的解决方案,凭借多年的经验,让设计与互联网技术结合,创造个性化、用户体验好的作品,建站类型包括:网站制作、成都网站建设、企业官网、英文网站、手机端网站、网站推广、域名注册、网站空间、企业邮箱。业务覆盖桥西地区。
那么,对于函数,它也是存放在内存块中的一段数据。例如下面的函数:
void func( int a )
{
printf( "in func, a = %d " , a );
}
此时,定义了一个函数名是func的函数。可以如下调用该函数:
func(100);
此时,就进入了func函数的函数体中执行。可以看到, 函数名如同数组名一样,代表函数所在内存块的首地址 。通过数组名可以访问数组在内存块中申请的内存,同理,通过函数名,可以访问函数在内存中存放的数据。
所以,函数名就代表了该函数在内存块中存放的首地址。那么,函数名是表示一个地址,就可以把这个地址值存放在某一个指针变量中,然后,通过指针变量访问函数名指向的函数。
在C语言中,提供了函数指针变量,可以存放函数名表示的地址。函数指针变量的定义格式如下:
返回数据类型 (*函数指针变量名)(形参列表)
对比函数的定义如下:
返回数据类型 函数名(形参列表)
可以看到,函数指针变量的定义,与函数的定义格式基本一样,唯一的区别是把“函数名”转换为“*(函数指针变量名)”;总结如下:
(1) 使用指针降级运算符*来定义,表示这个是一个指针。
(2) 指针降级运算符*不可以靠近返回数据类型,例如“返回数据类*”就表示函数的返回类型是一个指针。那么,为了让指针降级运算符*能够修饰函数指针变量,就用小括号()把指针降级运算符*与函数指针变量名包含起来。
定义了函数指针变量之后,可以把函数名赋给函数指针变量。因为,函数名就表示函数在内存块中的首地址,所以,可以直接把一个地址赋值给函数指针变量。格式如下:
函数指针变量 = 函数名;
最终,可以通过函数指针变量调用函数,调用的格式与通过函数名调用完全一样,通过函数指针变量调用函数,有如下形式:
方法1:函数指针变量(实参列表);
方法2:(*函数指针变量名)(实参列表);
很多情况下,我们更倾向于使用第一种形式,因为,它的使用方式更接近于通过函数名调用函数。
下面根据程序测试例子来看看怎么样应用函数指针变量。
深入学习,可以交个朋友,工人人人号:韦凯峰linux编程学堂
程序运行结果如下:
深入学习,可以交个朋友,工人人人号:韦凯峰linux编程学堂
可以看到,我们定义了func函数和函数指针变量pfunc,然后,把函数名func设置给函数指针变量pfunc,最终,通过函数指针变量pfunc调用函数。
因为函数指针变量存放的就是函数名表示的地址,所以,函数指针变量与函数名一样,可以直接通过函数指针变量调用函数。
注意:我们在学习指针的时候,可以把一个int类型的变量地址赋值给int类型的指针;但是,不可以把int类型变量的地址,赋值给double类型的指针。这就是变量数据类型不一致的问题。
同样的道理,定义函数的时候,函数的返回数据类型和形参列表都不一样,所以,函数指针变量能够接收的函数名,它们定义的 函数返回数据类型和形参列表必须一致 ,此时,就如同变量与指针变量类型一致时,才可以把变量的地址赋值给指针变量一样。
如下是一个测试例子:
深入学习,可以交个朋友,工人人人号:韦凯峰linux编程学堂
程序编译结果如下:
深入学习,可以交个朋友,工人人人号:韦凯峰linux编程学堂
可以看到,我们把func函数的形参列表修改为double,但是,函数指针变量pfunc定义的形参列表为int类型,此时,函数和函数指针变量的定义格式不一致,所以,不可以把函数名表示的地址设置给函数指针变量。我们来总结一下:
(1) 在Ubuntu系统中,使用GCC编译,提示warning警告,但是,程序可以编译通过,可以运行。
(2) 在Windows系统中,使用Visual Studio工具,无法编译该代码,提示类型不一致。
(3) 从代码的严谨方面来说,是不可以设置类型不一致的数据。所以,我们应该编写严谨的代码,函数定义的类型,与函数指针类型不一致的时候,不可以把函数名,赋值给函数指针变量。
函数指针变量的定义很重要,我们需要牢记和理解它们使用的方式。下面多举几个例子说明函数指针变量的定义和使用。
int func( void );
int (*pfunc)( void );
pfunc = func;
此时,定义func函数,它的返回值类型是int类型,形参列表是void,那么,定义pfunc函数指针变量的时候,它的返回值类型与形参列表都必须与func一样。
char * func1( int x, int y, int x);
char * (*pfunc1)( int , int , int );
pfunc1 = func1;
char * (*pfunc1)( int x, int y, int x);
我们再总结一下:
(1) 函数名表示函数在内存块中的首地址,可以直接把函数名赋值给函数指针变量;
(2) 定义函数指针变量的时候,函数返回数据类型和形参列表必须与要指向函数的定义一致;
供参考:
//:
strcpyDemo.c
//author:
Problue
#include
stdio.h
#include
assert.h
/**
*
该函数将从src指向的数组中复制最多count个字符(不复制空字符后面的)到dest指向的数组中。
*
如果复制发生在两个重叠的对象中,则行为未定义。
*
如果src指向的数组是一个长度比count短的字符串,则在dest指向的数组
*
后面添加空字符.直到写入了count个字符。
*@param
dest
目标数组
*@param
src
源数组
*@return
dest
*
*/
char*
strcopy(char*
dest,
const
char*
src,
int
count)
{
char*
sc
=
dest;
assert
((dest
!=
NULL)
(src
!=
NULL));
while
(
count
*src
!=
'\0')//
复制字符
[max
count]
src
到
dest
[count]
{
*sc++
=
*src++;
--count;
}
for
(;
count;
--count)
//
写入空字符到dest
{
*sc
=
'\0';
}
return
dest;
}
typedef
char*
(*STR_FUNC)(char*,
const
char*,
int);
int
main()
{
STR_FUNC
func
=
strcopy;
const
int
A_LENGTH
=
30;
char
a[A_LENGTH
+
1];//
=
"abc";
char*
b
=
"defg";
func(a,
b,
A_LENGTH);
puts(a);
puts(b);
getchar();
return
0;
}
functionpointer就是函数指针,指向一个函数,该函数的原型类似
void function(pStruct_X * p)
P.functionpointer,就等于是调用该函数了。
不过你的代码中,没写初始化,到底这个函数指针是指向哪个函数。
void* f(int); //一个指针函数f
void*(*p)(int) = f; //一个指向f的函数指针p
是的。函数指针就是函数的指针,它是一个指针,指向一个函数。这正如用指针变量可指向整型变量、字符型、数组一样,这里是指向函数。
例:
A) char * (*fun1)(char * p1,char * p2);
B) char * *fun2(char * p1,char * p2);
表达式相比,唯一不同的就是函数的返回值类型为char**,是个二级指针。
C) char * fun3(char * p1,char * p2);
fun3是函数名,p1,p2是参数,其类型为char *型,函数的返回值为char *类型。
扩展资料
函数指针使用的例子
#include stdio.h
#include string.h
char * fun(char * p1,char * p2)
{
int i = 0;
i = strcmp(p1,p2);
if (0 == i)
{
return p1;
}
else
{
return p2;
}
}
int main()
{
char * (*pf)(char * p1,char * p2);
pf = fun;
(*pf) ("aa","bb");
return 0;
}