十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
可以在函数内使用,请看下面的解释,希望对你有帮助。
怀宁网站建设公司创新互联,怀宁网站设计制作,有大型网站制作公司丰富经验。已为怀宁上千多家提供企业网站建设服务。企业网站搭建\成都外贸网站制作要多少钱,请找那个售后服务好的怀宁做网站的公司定做!
函数的参数分为形参和实参两种。
形参出现在函数定义中,在整个函数体内都可以使用,离开该函数则不能使用。
形参和实参的功能是数据传送,发生函数调用时,主调函数把实参的值传送给被调函数的形参从而实现主调函数向被调函数的数据传送。
※※※※※※※※※※※第一个问题的答复※※※※※※※※※※※※※※※
看明白了你的意思,
其实不知道你自己有没有注意到,你所尝试的方法,
如果用C语言来做的话,其实就是实现了C语言的部分面向对象的实现,
说是"部分"的原因是,这仅仅是实现了面向对象的“方法”。
如果想实现的话,准确的讲,应该不是你所说的,将"函数"作为形参,
应该是将“函数指针”作为形参。
这个在回调(CallBack)函数设计时,使用的非常多,
简单举一个例子:
#include stdlib.h
#include stdio.h
int Do1()
{
return 0;
}
int Do2(int num)
{
printf("The num is: %d\n", num);
return 0;
}
void CallBack1(void (*ptr)())//指向函数的指针作函数参数
{
(*ptr)();
}
void CallBack2(int n, int (*ptr)())//指向函数的指针作函数参数,这里第一个参数是为指向函数的指针服务的,
{ //不能写成void Caller2(int (*ptr)(int n)),这样的定义语法错误。
(*ptr)(n);
return;
}
int main()
{
CallBack1(Do1); //相当于调用Do1();
CallBack2(50, Do2); //相当于调用Do2(50);
return 0;
}
※※※※※※※※※※※第一个问题的答复※※※※※※※※※※※※※※※
※※※※※※※※※※※补充问题的答复※※※※※※※※※※※※※※※
针对你的补充问题,解答如下:
这个是可变形参的实现,准确地说,不是通过数组实现的,而是通过栈实现的。
C语言中的printf,scanf就是最常见的可变形参函数,定义一个可变形参的函数很简单,如void print(int n, ...) ,函数中对参数的处理主要是通过对栈进行操作,而c函数的实参都是自右向左压入栈的. 主要的栈操作(都是宏)有va_list,va_start ,va_arg,va_end, 定义如下:
typedef char * va_list;
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) ~(sizeof(int) - 1) )
#define va_start _crt_va_start
#define va_arg _crt_va_arg
#define va_end _crt_va_end
#define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define _crt_va_end(ap) ( ap = (va_list)0 )
va_start(ap,v):主要是获取可变参数列表的首地址,然后赋值给ap,近似ap=v+sizeof(v) (这里暂不考虑内存对齐和类型转换)
va_arg(ap,t):取得返回类型t的可变参数值, 并使ap指向下一个参数: ap += sizeof(t),这里的t是可变参数的数据类型,如int,float之类
va_end(ap):给ap初始化
va_start(ap,v) va_arg(ap,t) va_end(ap)三者合用,保证程序的健壮性.
一个使用可变形参的简单程序:
#include stdio.h
#include stdarg.h //包含va_list等定义
float sum( float first, ... ) //,...代表可变形参函数
{
float i=first,sum=0;
va_list maker; //va_list 类型数据可以保存函数的所有参数,做为一个列表一样保存
va_start(maker,first); //设置列表的起始位置
while(i!=-1.0)
{
sum+=i;
i=va_arg(maker,float); //返回maker列表的当前值,并指向列表的下一个位置
}
return sum;
}
void main(void)
{
printf( "sum is: %f\n", sum( 2.0,8.0,8.5,-1.0 ) ); //函数调用
}
※※※※※※※※※※※补充问题的答复※※※※※※※※※※※※※※※
主函数的实参是由操作系统传递的。
c++规定的主函数标准格式(c语言未具体规定):int
main(int
argc,
char
*argv[])
其中int
argc
表示操作系统传递给程序的参数个数,char
*argv[]
存储各个参数。
例如:
若有程序如下:
int
main(int
argc,
char
*argv[])
{
int
i;
printf("您输入了
%d
个参数\n分别是:",argc);
for(i=0;i
运行-
cmd
-
输入
c:\app.exe
hello
world!
回车
执行结果将是:
您输入了
3
个参数
分别是:c:\app.exe
hello
world!
"
注意,第一个参数值将是你程序的文件名。
不知道函数调用原理的话,很难说得清楚的。粗略的说在C中函数其实是分成三个部分,分别是声明、实现和使用。声明就是告诉C有这么个函数,实现就是这个函数的完整代码,使用则是调用这个函数帮你做运算或者是做事情。为了让函数能做更多的事情,就需要一种方法能向函数传递性息,最可靠、最方便的方法就是参数传递,C就是这样做的。参数传递也不是简单的一件事情,首先在调用者和被调用者之间得有个约定,大家约定好了参数放什么地方,如何使用,这个就称为调用约定。现在我们回过头来看什么叫实参和形参。由上面的叙述可以看到,函数其实有一定的独立性的,参数是在调用约定下放在指定地方供函数使用的,在我们实现函数的时候,实际的参数是什么我们还不知道,但我们知道有多少参数,分别是什么数据类型的和放在什么地方(顺序),这个是由调用约定来保证的。我们知道的这些就是形式的或者说是逻辑的,我们在实现函数(编写函数体)的时候需要给他们命名,以便使用,这些在函数实现的时候命名的形式上的参数就是形式参数,简称形参。而在实际调用发生的时候,调用者会将实际的参数放在约定好了的地方,这个实际参数简称实参。函数被运行的时候实参和形参形成了一一对应,对形参的操作就变成了对实参的操作。
对调用约定还有需要说明的就是,现在常用的调用约定都是传值的,也就是说只是传递了一个副本,这样做的结果就是你对形参的操作不会被传递出来。
是否可以解决您的问题?
形参:全称为“形式参数”是在定义函数名和函数体的时候使用的参数,目的是用来接收调用该函数时传递的参数。
形参的作用是实现主调函数与被调函数之间的联系,通常将函数所处理的数据,影响函数功能的因素或者函数处理的结果作为形参。
实参:全称为"实际参数"是在调用时传递给函数的参数,即传递给被调用函数的值。
实参可以是常量、变量、表达式、函数等, 无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值, 以便把这些值传送给形参。 因此应预先用赋值,输入等办法使实参获得确定值。
扩展资料:
形参和实参的特点:
1、形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只在函数内部有效。函数调用结束返回主调用函数后则不能再使用该形参变量。
2、实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使参数获得确定值。
3、实参和形参在数量上,类型上、顺序上应严格一致,否则就会发生类型不匹配的错误。
4、在一般传值调用的机制中只能把实参传送给形参,而不能把形参的值反向地传送给实参。因此在函数调用过程中,形参值发生改变,而实参中的值不会变化。而在引用调用的机制当中是将实参引用的地址传递给了形参,所以任何发生在形参上的改变实际上也发生在实参变量上。
参考资料来源:百度百科-形参
参考资料来源:百度百科-实参