函数重载
重载函数通常用来命名一组功能相似的函数,这样做减少了函数名的数量,避免了名字空间的污染。这些同名函数的形式参数(指参数的个数、类型或者顺序)必须不同
构成函数重载的条件如下:
1、函数名相同。
2、这些同名函数的形式参数(指参数的个数、类型或者顺序)必须不同,即参数列表不同。
注意,返回值类型不同不构成重载函数。
重载函数的匹配原则:
1、严格匹配,找到后就调用。
2、没有找到严格匹配的函数,但是可以通过隐式转化,找到适合的重载函数。
注意:int到long和double,double到int和float的隐式转换,会引发二义性。
define的“函数定义”
define可以像函数那样接受一些参数,如#define max(x,y) (x)>(y)?(x):(y);
这个定义就将返回两个数中较大的那个。
#define pin (int*);pin a,b;
本意是a和b都是int型指针,但是实际上变成int* a,b。
#define va_list char* /* 可变参数地址 */
#define va_start(ap, x) ap=(char*)&x+sizeof(x) /* 初始化指针指向第一个可变参数 */
#define va_arg(ap, t) (ap+=sizeof(t),*((t*)(ap-sizeof(t)))) /* 取得参数值,同时移动指针指向后续参数 */
#define va_end(ap) ap=0 /* 结束参数处理 */
参数是通过堆栈传递的,参数按照从右向左的顺序入栈。
1. va_start。为了得到第一个可选参数的地址,我们有三种办法可以做到:
A) = &n3 + _INTSIZEOF(n3)
// 最后一个固定参数的地址 + 该参数占用内存的大小
B) = &n2 + _INTSIZEOF(n3) + _INTSIZEOF(n2)
// 中间某个固定参数的地址 + 该参数之后所有固定参数占用的内存大小之和
C) = &n1 + _INTSIZEOF(n3) + _INTSIZEOF(n2) + _INTSIZEOF(n1)
// 第一个固定参数的地址 + 所有固定参数占用的内存大小之和
2. va_arg身兼二职:返回当前参数,并使参数指针指向下一个参数。
初看va_arg宏定义很别扭,如果把它拆成两个语句,可以很清楚地看出它完成的两个职责。
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) //下一个参数地址
// 将( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )拆成:
/* 指针ap指向下一个参数的地址 */
1. ap += _INTSIZEOF(t); // 当前,ap已经指向下一个参数了
/* ap减去当前参数的大小得到当前参数的地址,再强制类型转换后返回它的值 */
2. return *(t *)( ap - _INTSIZEOF(t))