C++函数完全兼容C函数定义的风格,但是也做出了一些改进。一方面,C++函数允许使用缺省参数值和占位参数;另一方面,C++提供了重要的函数重载机制;另外,为了解决C中臭名昭著的宏缺陷问题,C++提供了内联函数的机制(C99之后也支持这个特性)。从C++11开始,函数定义支持后置返回类型。
一、缺省参数值
C++中可以在函数声明时为参数提供一个默认值,当函数调用时如果不提供实参,就使用这个默认值:
int func(int x = 0);
int main(int argc, char *argv[])
{
printf("func() = %d\n", func());
return 0;
}
int func(int x)
{
return x;
}
注意:默认值只在函数声明时有效,虽然函数定义中也可以写默认值,但是会被函数声明中的默认值覆盖。
函数默认参数的规则是:
- 参数的默认值需要从右向左提供
- 函数调用时如果使用了默认值,那么后续的参数都必须使用默认值
二、函数的占位参数
在C++中可以为函数提供占位参数。占位参数只有参数类型声明,但是没有参数名;这样,在函数的实现中是无法使用这个参数的:
int func(int)
{
return 0;
}
int main(int argc, char *argv[])
{
func(1); // 虽然参数无意义,但是还是要提供来通过编译。
return 0;
}
占位参数存在的意义是:兼容C语言中可能出现的不规范的写法,也可以配合默认值来一起使用。
三、函数重载
如果同一个作用域内的几个函数名称相同,但是参数列表不同,那它们就是重载函数:
void print(const char *cp);
void print(const int *begin, const int *end);
void print(const int ia[], size_t size);
函数重载有以下几个注意点:
(3.1) 函数重载必须至少要满足的条件
- 参数个数不同
- 参数类型不同
- 参数顺序不同
注意:返回值类型不同不能作为重载的条件。
(3.2) const与重载
如果函数以值传递参数,那么const
不能构成重载:
void func(int arg);
void func(const int arg); // 错误:对func的重定义
但是,如果以指针或者引用来传递参数,那么const就可以构成重载了:
void func1(int *arg);
void func1(const int *arg); // 正确:声明了新函数
void func1(const int *const arg); // 正确:声明了新函数
void func2(int &arg);
void func2(const int &arg); // 正确:声明了新函数
另一方面,const成员函数
和非const成员函数
同样可以构成函数重载:
class object
{
public:
int func();
const int func() const; // 正确:声明了新函数
};
(3.3) 重载函数的确定
编译器会以下面的顺序确定要调用的函数:
- 精确匹配:
- 实参的类型和形参完全相同。
- 实参从数组类型或者函数类型转换为对应的指针类型。
- 向实参添加
顶层const
或者从实参中移除顶层const
- 通过
const
转换实现的类型匹配。 - 通过类型提升实现的匹配。
- 通过算数类型转换实现的匹配。
- 通过类类型转换实现的匹配。
(3.4) 重载与函数指针
由于编译器需要根据重载规则去挑选与函数指针参数列表一致的函数,并且要严格地匹配函数类型与函数指针的类型,因此无法直接通过函数名得到重载函数的入口地址:
void func(int);
void func(double);
int main(int argc, char *argv[])
{
void * v = func // 错误:无法通过函数名得到函数的地址。
void(*pFunc1)(int) = func; // 正确:获得了void func(int)的入口地址。
void(*pFunc2)(double) = func; // 正确:获得了void func(double)的入口地址。
return 0;
}
(3.5) 让编译器以C语言方式编译函数
为了兼容旧有的C语言代码库,必须以C语言的编译规则来编译函数,因此需要使用如下的方式:
- 以
__cplusplus
宏来检查是否使用了C++ - 以
extern "C"
来让编译器以C语言方式编译函数
#ifdef __cplusplus
extern "C" {
#endif
void func(); // 这个函数将以C语言的方式编译。
#ifdef __cplusplus
}
#endif
四、内联函数
C++中推荐以内联函数来代替宏代码片段。C++同样以直接替换代码块的方式来处理内联函数,同时没有著名的宏缺陷问题。可以使用inline
关键字来请求编译器将函数以内联函数的方式处理 (编译器可能忽略这个请求):
inline void func() {...}
注意:
内联函数声明时
inline
必须和函数定义结合在一起,否则编译器会忽略该请求。- 现代的编译器会进行优化,一些函数即使没有
inline
,也可能会被内联编译。 - 一些编译器提供了一些扩展语法,可以对函数进行强制的内联操作:
- VC:
__forceinline
- g++:
__attribute__((always_inline))
- VC:
内联函数存在一些限制:
- 不能存在循环语句
- 不能存在过多的条件判断语句
- 函数体不能太长
- 不能取函数的地址
- 内敛函数声明必须在调用这个函数之前
五、后置返回类型
C++11提供了函数的后置返回类型,用于兼容自动类型推断:
auto func() -> void {} // 等价于 void func() {}