1.C++中的函数重载
- 重载:同一标志符在不同的上下文有不同的意义
- 函数重载(Function Overload) :用同—个函数名定义不同的函数
- 当函数名和不同的参数搭配时函数的含义不同
- 函数重载至少满足下面的一个条件:
- 参数个数不同
- 参数类型不同
- 参数顺序不同
#include <stdio.h>
#include <string.h>
int func(int x)
{
return x;
}
int func(int a, int b)
{
return a + b;
}
int func(const char* s)
{
return strlen(s);
}
int main(int argc, char *argv[])
{
printf("%d\n", func(3)); // 3
printf("%d\n", func(4, 5)); // 9
printf("%d\n", func("D.T.Software")); // 12
getchar();
return 0;
}
- 当函数默认参数遇上函数重载会发生什么?
- C++引入了太多特性,然而有些特性之间会产生冲突
#include <stdio.h>
#include <string.h>
int func(int a, int b, int c = 0)
{
return a * b * c;
}
int func(int a, int b)
{
return a + b;
}
int main(int argc, char *argv[])
{
int c = 0;
c = func(1, 2); // 存在二义性,调用失败,编译不能通过
printf("c = %d\n", c);
printf("Press enter to continue ...");
getchar();
return 0;
}
- 编译器调用重载函数的准则
- 将所有同名函数作为候选者
- 尝试寻找可行的候选函数
- 精确匹配实参
- 通过默认参数能够匹配实参
- 通过默认类型转换匹配实参
- 匹配失败
- 最终寻找到的候选函数不唯—,则出现二义性,编译失败。
- 无法匹配所有候选者,函数未定义,编译失败。
- 函数重载的注意事项
- 重载函数在本质上是相互独立的不同函数
- 重载函数的函数类型不同
- 函数返回值不能作为函数重载的依据
- 函数重载是由函数名和参数列表决定的!
- 重载与指针
- 下面的函数指针将保存哪个函数的地址?
- 函数重载遇上函数指针
- 将重载函数名赋值给函数指针时:
- 1. 根据重载规则挑选与函数指针参数列表—致的候选者
- 2. 严格匹配候选者的函数类型与函数指针的函数类型
- 将重载函数名赋值给函数指针时:
#include <stdio.h>
#include <string.h>
int func(int x) // int(int a)
{
return x;
}
int func(int a, int b)
{
return a + b;
}
int func(const char* s)
{
return strlen(s);
}
typedef int (*PFUNC)(int a); // 定义一个函数指针,int(int a)
int main(int argc, char *argv[])
{
int c = 0;
PFUNC p = func;
c = p(1);
printf("c = %d\n", c); // 1
printf("Press enter to continue ...");
getchar();
return 0;
}
- 可以通过函数指针类型获得重载函数的地址,如:int( * )(int, int, int)
- 注意
- 函数重载必然发生在同—个作用域中
- 编译器需要用参数列表或函数类型进行函数选择
- 无法直接通过函数名得到重载函数的入口地址
2.C++和C相互调用
- 实际工程中C++和C代码相互调用是不可避免的
- C++编译器能够兼容C语言的编译方式
- C++编译器会优先使用C++编译的方式
- extern关键字能强制让C++编译器进行C方式的编译
2.1 C++调用C函数
- add.h
int add(int a, int b);
- add.c
#include "add.h"
int add(int a, int b)
{
return a + b;
}
- main.cpp
#include <stdio.h>
extern "C"
{
#include "add.h"
}
int main()
{
printf("1 + 2 = %d\n", add(1, 2)); // 3
return 0;
}
2.1 C调用C++函数
- add.h
int add(int a, int b);
- add.cpp
extern "C"
{
#include "add.h"
int add(int a, int b)
{
return a + b;
}
}
- main.c
#include <stdio.h>
#include "add.h"
int main()
{
printf("1 + 2 = %d\n", add(1, 2));
getchar();
return 0;
}
2.3 问题
- 如何保证—段C代码只会以C的方式披编译?
- 解决方案
- _cplusplus是C++编译器内置的标准宏定义
- __cplusplus的意义:确保C代码以统—的C方式被编译成目标文件
#ifdef __cplusplus
extern "C"
{
#endif
//C-Style Compilation
#ifdef __cplusplus
}
#endif
2.4注意事项
- C++编译器不能以C的方式编译重载函数
- 编译方式决定函数名被编译后的目标名
- C++编译方式将函数名和参数列表编译成目标名
- C编译方式只将函数名作为目标名进行编译
#include <stdio.h>
#include <string.h>
#ifdef __cplusplus
extern "C" {
#endif
int func(int a, int b)
{
return a + b;
}
int func(const char* s)
{
return strlen(s);
}
#ifdef __cplusplus
}
#endif
int main(int argc, char *argv[])
{
printf("Press enter to continue ...");
getchar();
return 0;
}
- error C2733: “func”: 不允许重载函数的第二个 C 链接
3.小结
- 函数重载是C++对C语言的一个重要升级
- 函数重载通过函数参数列表区分不同的同名函数
- 函数的返回值类型不是函数重载的依据
- extern关键的关键能够实现C和C++的相互调用