partone---第六章 函数

来源:互联网 发布:立石和 知乎 编辑:程序博客网 时间:2024/06/05 18:09
fig1.jpg
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 &);

  1. 该函数的函数类型为bool lengthCompare(const string &,const string &)。 函数类型,是由形参列表和返回值决定,函数名称只是函数的别名而已。
  2. 声明一个函数指针,只需要使用指针代替函数即可(因为函数名称仅仅是一个别名)
    bool (*pf) (const string &,const string &)
  3. 如何使用函数指针
    当把函数名作为一个值使用时,该函数会自动的转换为指针。
    指向不同的函数指针之间,是不能转换的,因为代表的函数返回值。
pf = lengthCompare;   // 指向名为lengthCompare的函数pf = &lengthCompare; //等价,取址操作符是可选的。//直接使用bool b1 = pf("hello", "world");bool b2 = (*pf) ("hello", "world"); //等价调用
  1. 函数指针形参
//第三个形参是函数类型,自动转换为指向函数的指针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));