有默认参数的函数:
一般情况下,在函数调用时形参从实参那里取得值,因此实参的个数应与形参相同。当用同样的实参多次调用同一函数时,C++提供了简单的处理办法,给形参一个默认值,这样形参就不必一定要从实参取值了。如有一函数声明
float area(float r=6.5);
指定r的默认值为6.5,如果在调用此函数时,确认r的值为6.5,则可以不必给出实参的值,如
area(); //相当于area(6.5);
如果不想使形参取此默认值,则通过实参另行给出。如
area(7.5); //形参得到的值为7.5,而不是6.5
这种方法比较灵活,可以简化编程,提高运行效率。
如果有多个形参,可以使每个形参有一个默认值,也可以只对一部分形参指定默认值,另一部分形参不指定默认值。如有一个求圆柱体体积的函数,形参h代表圆柱体的高,r为圆柱体半径。函数原型如下:
float volume(float h,float r=12.5); //只对形参指定默认值12.5
函数调用可以采用以下形式:
volume(45.6); //相当于volume(45.6,12.5)
volume(34.2,10.4) //h的值为34.2,r的值为10.4
实参与形参的结合是从左至右顺序进行的,第1个实参必然与第1个形参结合,第2个实参必然与第2个形参结合……因此指定默认值的参数必须放在形参表列中的最右端,否则出错。例如:
void fl(float a, int b=0,int c, char d='a'); //不正确
void f2(float a, int c,int b=0, char d=y'); //正确
如果调用上面的f2函数,可以采取下面的形式:
f2(3.5,5,3,'x') //形参的值全部从实参得到
f2(3.5,5,3) //最后一个形参的值取默认值
f2(3.5, 5) //最后两个形参的值取默认值,b=0,d='a'
可以看到,在调用有默认参数的函数时,实参的个数可以与形参不同,实参未给定的,从形参的默认值得到值。利用这一特性,可以使函数的使用更加灵活。
编写程序:
运行结果:
程序分析:
如果想从3个数中找大者,可以在调用时写成"max(a,b,c)"形式;如果只想从两二个正整数中找大者,则在调用时写成"max(a,b)"形式,此时c自动取默认值0,由于0比任何正整数都小,因此从14,-56,0中选最大数和从14,-56中选大者的结果是一样的。
在使用带有默认参数的函数时有两点要注意:
(1)如果函数的定义在函数调用之前,则应在函数定义中给出默认值。如果函数的定义在函数调用之后,则在函数调用之前需要有函数声明,此时必须在函数声明中给出默认值,在函数定义时可以不给出默认值,也就是说必须在函数调用之前将默认值的信息通知编译系统。由于编译是从上到下逐行进行的,如果在函数调用之前未得到默认值信息,在编译到函数调用时,就会认为实参个数与形参个数不匹配而报错。
如果在声明函数时已对形参给出了默认值,而在定义函数时又对形参给出默认值,有的编译系统会给出"重复指定默认值"的报错信息,有的编译系统对此不报错,甚至允许在声明时和定义时给出的默认值不同,此时编译系统以先遇到的为准。由于函数声明在函数定义之前,因此以声明时给出的默认值为准,而忽略定义函数时给出的默认值。例如在函数声明时指定c的默认值为﹣32767,而在定义函数时指定c的默认值为123,则编译系统取c的默认值为﹣32767。为了避免混淆,最好只在函数声明时指定默认值。
(2)一个函数不能既作为重载函数,又作为有默认参数的函数。因为当调用函数时如果少写一个参数,系统无法判定是利用重载函数还是利用默认参数的函数,出现二义性,系统无法执行。
说明:内置函数、函数的重载、函数模板和有默认参数的函数在C语言中是没有的,是C++增加的。读者在学习时,先对它们有一般的了解,在基于对象的程序设计中会用到这些方法,以提高编程的质量和效率。