版权声明:自由转载-非商用-非衍生-保持署名 | Creative Commons BY-NC-ND 4.0
跟变量一样,函数也需要先声明再使用。同样的,函数的定义和函数的声明也可以分离:一个函数只能定义一次,但可以声明多次。
函数声明必须包含:返回类型、函数名和形参列表。形参列表必须包含形参类型,但没强调要包含形参名。比如
1  | void func(int, string);  | 
默认实参
调用函数时,可以省略有默认值的实参。如果一个参数有默认值,那么它后面的实参都必须有默认值。如:
1  | string screenInit(string::size_type height = 24,  | 
设置默认形参时,最少使用的默认值放在最前,最多使用的放在最后。
函数在声明时可以指定默认实参,但是在一个文件中,只能为形参指定一次默认参数。所以可以将声明放在头文件中。
静态局部对象
上面的形参都会在定义它们的快语句结束时被撤销,如果需要跨越多个作用域,定义为 static,它能保证在程序结束前不被撤销:
1  | size_t count() {  | 
内联函数
一般的函数调用要比求解表达式慢得多,内联函数可以避免函数调用的开销。使用 inline 关键字定义,如:
1  | inline const string &shorterString(const string &s1, const string &s2)  | 
则在调用 cout << shorterString(s1, s2) << endl; 时, 编译时展开为 cout << (s1.size() > s2.size() ? s1 : s2) << endl;。
内联机制适用于优化小的、几行的且经常调用的函数。
类的成员函数
和任何函数一样,包含下面四个部分:
- 返回类型
 - 函数名
 - 逗号分隔的形参列表(可为空)
 - 花括号内的函数体
 
1,2,3 组成的是函数原型,函数的原型必须在类中定义,函数体则可以在类外。假设 Sales_item 类有两个成员函数 avgPrice() 和 sameIsbn(const Sales_item &):
1  | class Sales_item {  | 
可以发现形参的后面有 const,在解释之前,说明成员函数如何定义:
成员函数的函数体
类的所有成员必须在定义类的花括号中声明,并且编译器隐式地将类内定义的成员函数当作是内联函数。
类的成员函数可以访问 private 成员,如 isbn。
this 指针的引入
每个成员函数都有一个额外的、隐式的形参 this,在调用成员函数时,形参 this 初始化为调用函数的对象的地址。可以这样理解:
1  | item.sameIsbn(otherItem);  | 
编译器会重写这个函数调用:
1  | Sales_item::sameIsbn(&item, otherItem);  | 
const 成员函数的引入
也就是上面提到的形参后面 const,它改变了隐式形参 this 的类型。
1  | bool sameIsbn(const Sales_item *const this,  | 
这种使用 const 的函数称为常量成员函数(const member function)。由于 this 指向 const 对象,const 成员函数不能修改调用该函数的对象。
类外定义成员函数
1  | double Sales_item::avgPrice() const {  | 
使用了域操作符 :: 及类名 Sales_item。
Sale_item 的构造函数
构造函数是特殊的成员函数,构造函数名与类名相同,而且无返回类型。可以有多个构造函数,相互之间的具有不同数目或类型的形参。
跟普通成员函数一样,必须在类内声明,类内或类外定义。
1  | class Sales_item {  | 
构造函数的初始化列表
在冒号与花括号之间的代码称为构造函数的初始化列表,即 units_sold(0), revenue(0.0) 为成员指定初值,括号内是初值。构造函数的形参表为空说明此为默认调用的
类代码的组织
通常将类的声明放置在头文件中,在类外定义的成员函数则置于源文件中。
重载函数
在相同作用域中,出现相同名字而形参表不同的函数,称为重载函数。如电话本的查找:基于姓名、基于号码
1  | Record lookup(const Name&);  | 
重载与重复声明的区别
如果两个函数声明的返回类型和参数表完全匹配,叫重复声明;如果形参表完全相同,返回类型不同,则第二个错误。
在函数中局部声明的名字会屏蔽全局名,即使是变量名对函数名也同样成立:
1  | string init();  | 
指向函数的指针
函数指针是指指向函数而非对象的指针,像其他指针一样,函数指针指向的是函数类型,函数类型由其返回类型和形参表确定,与函数名无关。
1  | bool (*pf)(const string &, const string &);  | 
用typedef简化定义
1  | typedef bool (*cmpFcn)(const string &, const string &);  | 
则 cmpFcn 表示指向返回类型为 bool 类型并带有两个 const string 引用形参的函数的指针。若要定义此类型的函数指针,则直接使用 cmpFcn 即可。
初始化和赋值
在引用函数名又没调用该函数,函数名将被自动解释为指向函数的指针。假设有函数:
1  | bool lengthCpmpare(const string &, const string &);  | 
会被解释为如下类型的指针:
1  | bool (*)(const string &, const string &);  | 
可以做如下初始化:
1  | cmpFcn pf1 = 0; // ok:不指向任何函数  | 
此时,直接饮用函数名等效于在函数名上应用取地址操作符:
1  | cmpFcn pf1 = lengthCompare;  | 
函数指针只能通过同类型的函数或函数指针或 0 值常量表达式进行初始化或赋值
通过指针调用函数
不需要解引用操作符,直接通过指针调用函数:
1  | cmpFcn pf = lengthcompare;  | 
函数指针形参
函数的形参可以是指向函数的指针,这种形参可以用以下两种形式编写:
1  | void useBigger(const string &, const string &,  | 
返回指向函数的指针
函数可以返回指向函数的指针,但是,这并不简单:
1  | // ff is a function taking an int and returing a function pointer  | 
这样理解: ff(int) 表明 ff 为一个函数,它带有一个 int 型形参,该参数返回 int (*)(int*, int),它是指向一个函数的指针。用 typedef 可以更加明白:
1  | // PF is a pointer to a function returing an int, taking an int* and an int  | 
END.