您的位置:首页 > 其它

partone---第六章 函数

2017-10-25 23:11 225 查看


fig1.jpg

函数是一个命名了的代码块,可有0个或多个参数。


6.1 函数基础


局部对象

自动对象: 存在于块执行期间,当块执行完毕时,则内存消失。例如函数的形参。
局部静态对象: 生命周期为第一次定义到程序结束才终止。


函数声明

函数声明应该放到头文件中。 这样做的好处在于:同一函数的所有声明一致;想改变函数的接口,只需要改变函数的声明的即可。


6.2 参数传递

引用传递: 形参的类型决定了形参和实参的交互方式。 当形参是引用类型时,如其他引用一样,引用形参是对应实参的别名。
值传递: 将实参的值拷贝给形参,此时形参和实参是两个相互独立的对象。对形参的改变,不会影响到实参的值。


6.2.1 值传递

*指针形参
void reset(int *ip)
{
*ip=0;
ip=0;
}
int i=42;
reset(&i)


此时形参和实参是两个相互独立的对象。

实现两个数交换的两种方法(C++中,建议使用引用类型代替指针形参)
void swap1(int *a,int * b)//指针 参数是两个指针变量
{
int tmp;
tmp=*a; //把a指向的值赋给tmp
*a=*b;   //把b指向的值赋给a指向的值
*b=tmp;   //把tmp的值赋给b指向的值
//这样就达到了变换a,b指向的值的目的
}
void swap2(int &a,int &b)//引用 参数是两个整型变量的引用
{
//引用就是他本身的值,所以直接交换两个的值就行了。
int tmp;
tmp=a;
a=b;
b=tmp;
}


6.2.2 引用传递


使用引用避免拷贝


当拷贝大的类类型对象或容器对象时,比较低效。

使引用可返回额外信息


一个return只能返回一个参数,通过引用带出多个参数。


6.2.3 const形参和实参

void fcn(const int i){};
void fcn(int i){};


上面这两个函数是一样的,不能重载。因为当形参有顶层const时,传给它的是const或非const均可以。

顶层const:指针本身是一个常亮;

底层const:指针所指的对象是一个常量。
const  int ci=42;   //顶层const


指针或引用形参与const

非常量可以初始化一个底层const对象,反之则不行;

同时,一个普通类型的引用,只能用同类型的对象初始化。


尽量使用常量引用

顶层const作为形参时,实参传递是const或非const均可;
例如:
bool  is_empty(string &s)
{
return s.empty();
}

这种情况下,限制了该函数所能接受的实参类型,无法把const、对
象、字面值常量或者需要类型转换的对象传递给普通的引用形参。
另一方面,也可能带来误导,可修改字符串s。
应该把形参改为 const string &s


6.2.4 数组形参

数组的两个特性:不允许拷贝数组、数组名转换为指针


6.2.5 main:处理命令行选项

int main(int argc, char *argv[])
{
//其中argc是argv[][]的行数
}


6.2.6 含有可变形参的函数---initializer_lsit

initializer_lsit是一种标准库类型。适用于函数的实参未知,但类型都相同的情况下。
int icount(initializer_lsit<int> il)
{
int count =0;
for(auto val :il)
{
count += val;
}
return count;
}
icount({1,2,3,4,5});


6.3 返回类型和return语句


6.3.1 无返回值函数

函数的返回类型是void时,如果函数末尾没有return时,也可以。因为在函数的最后一句会隐式的调用return。


6.3.2 有返回值函数

return exp;

不要返回局部对象的引用或指针


函数终止后,局部变量的引用将不再指向有效的内存区域。

引用返回左值

char &get_val(string &str, int index)
{
return str[index];
}

void main
{
string s("a value");
get_val(s,0) = 'A';  //改为 A value
}


6.3.3 返回数组的指针

int (*func(int i))[10];


6.4 函数重载

重载函数:名字相同,但是函数参数列表不同。

注:返回类型不算。

重载和const形参


顶层const

(因为当形参有顶层const时,传给它的是const或非const均可以)。
(因为当形参有顶层const时,传给它的是const或非const均可以),所以下面两组函数是重复声明。
Record lookup(phone);
Record lookup(const phone);

Record lookup(phone*);
Record lookup(phone* const );


形参是指针或者引用。将是底层const,可通过指向的对象时常量对象还是非常量对象实现重载。

一方面,const不能转换为其他类型,只能将const转换为const;另一方面,相反的,非const可以转换为const,但是当同时存在非常量和const两个版本的形参时,编译器会优先选择非常量版本的函数,
Record lookup(Account &);
Record lookup(const Account &);

Record lookup(phone*);
Record lookup(phone* const );


6.5 特殊用途语言特性


6.5.1 默认实参

规定:一旦某个形参被赋予了默认值,那么它后面的所有形参都必须被赋予默认值。


6.5.2 内联函数和constexpr函数

将函数指定为内联函数,通常就是在每个调用点进行展开。

constexpr函数---暂时用不到。


6.5.3 调试帮助


assert预处理


如:assert(expr),如果表达式为真,什么都不做;如果表达式为假,则输出信息并终止程序。

DEBUG 预处理命令


如果你把代码夹在#ifdef DEBUG 和对应的 #endif 中间,那么这段代码只有在调试(DEBUG)下才会被编译。也就是说,如果你在RELEASE模式下,这些代码根本就不会存在于你的最终代码里头。
#include <iostream>
using namespace std;
#ifdef DEBUG
inline void msg()
{
cout<<"I'm testing";
}
#else
inline void msg() {}
#endif
int main()
{
msg();
return 0;
}


6.6 函数匹配

背景:有几个重载函数,当一个调用函数调用重载函数时,可能会发生二义性。

解决办法:寻求最佳匹配。


6.7 函数指针

bool lengthCompare(const string &,const string &);

该函数的函数类型为bool lengthCompare(const string &,const string &)。 函数类型,是由形参列表和返回值决定,函数名称只是函数的别名而已。
声明一个函数指针,只需要使用指针代替函数即可(因为函数名称仅仅是一个别名)
bool (*pf) (const string &,const string &)

如何使用函数指针

当把函数名作为一个值使用时,该函数会自动的转换为指针。

指向不同的函数指针之间,是不能转换的,因为代表的函数返回值。
pf = lengthCompare;   // 指向名为lengthCompare的函数
pf = &lengthCompare; //等价,取址操作符是可选的。

//直接使用
bool b1 = pf("hello", "world");
bool b2 = (*pf) ("hello", "world"); //等价调用

函数指针形参
//第三个形参是函数类型,自动转换为指向函数的指针
void useBigger(const string &, const string &, bool  pf (const string &,const string &));
|
void useBigger(const string &, const string &, bool  (*pf) (const string &,const string &)); //自动转换为函数指针


如上面,这样显得函数冗长。

通常:
typedef bool Func(const string &, const string &) ;

void useBigger(const string &, const string &, Func));
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: