[C++] 函数

函数基础

一个典型的函数定义包括以下部分:返回类型、函数名字、由0个或多个形参组成的列表以及函数体。其中形参以逗号隔开,形参列表位于一个圆括号之内,函数指向的操作在语句块内,也就是函数体。

函数调用

使用调用运算符来执行函数,调用运算符的形式是一对圆括号,它作用于一个表达式,该表达式是函数或者指向函数的指针,圆括号内是逗号间隔开的实参列表,我们用实参初始化函数的形参。

函数的调用完成了两项工作:1、用实参初始化函数对应的形参;2、将控制权转移给被调用函数。

执行函数的第一步是定义并初始化它的形参。

形参和实参

实参是形参的初始值,实参的类型必须与对应的形参类型匹配,且数量相同。

int fact(int);

fact(3);
// 正确

fact(3.14)
// 正确
// 3,14(double) ~ 3(int)

形参列表

1、每个形参必须有声明;

2、每个形参的声明必须单独书写;

3、每个形参不能同名;

返回值类型

1、函数的返回值类型不能是数组类型或函数类型

2、函数的返回值类型可以是指向数组指针(或引用)或指向函数的指针。

局部对象

名字有作用域,对象有生命周期

1、名字的作用域是程序文本的一部分,名字在其中可见

2、对象的生命周期是程序执行过程中该对象存在的一段时间

函数的形参也是局部变量。

同时局部变量还会隐藏在外层作用域中同名的其他所有声明。

局部变量的生命周期依赖于定义的方式(static、非static)

自动对象

只存在于块执行期间的对象称为自动对象。当块执行结束后,块中创建的自动对象的值就变成未定义的了。

局部静态对象

local static object:在程序执行路径第一次经过对象定义语句时初始化,并且直到程序终止才被销毁,在此期间即使对象所在的函数结束执行也不会对它有影响。

局部静态变量没有显式的初始化,它将执行值初始化,内置类型的局部静态变量初始化为0.

函数声明

函数只能定义一次,但可以声明多次。如果一个函数永远也不会用到,那么它可以只有声明没有定义

函数原型也就是函数声明,包括函数返回类型、函数名、形参类型

在头文件中声明函数,在源文件中定义函数。

分离式编译

在分离式编译中,如果我们改变了一个源文件,只需要重新编译这个改动的文件即可。

参数传递

每次调用函数时都会重新创建它的形参,并用传入的实参对形参进行初始化。

如果形参是引用类型,它将绑定到对应的实参上也就是引用传递,否则将实参的值拷贝后赋给形参。引用形参是它对应的实参的别名。

当实参的值被拷贝给形参时,形参和实参是两个互相独立的对象,这样的实参被值传递。

传值参数

值传递时,函数对形参做的所有操作都不会影响实参。

指针形参

指针的行为和其他非引用类型一样,当执行指针拷贝操作时,拷贝的是指针的值,拷贝之后,两个指针是不同的指针,因为指针使我们可以间接的访问它所指的对象,所以指针可以修改它所指对象的值。

传引用参数

引用的操作实际上是作用在引用所引的对象上。

通过使用引用形参,允许函数改变一个或多个实参的值

当调用含有引用形参的函数时,只需要直接传入对象而无须传递对象的地址

通过使用引用来避免拷贝,因为拷贝大的类类型对象或者容器对象比较低效,甚至有的类类型不支持拷贝操作。所以通过可以引用形参来访问该类型的对象,如果不需要改变引用的对象,则可以把形参定义成对常量的引用。

如果函数无须改变引用形参的值,最好将其声明为常量的引用

通过使用引用形参可以返回额外信息

如果想让函数返回多个值,有以下两种方法:

1、引用形参为我们一次返回多个结果提供了有效的途径

2、定义一个新的数据类型,让它包含需要返回的成员。

const形参与实参

如果形参是const时,需要考虑关于顶层const的讨论,顶层const保证对象本身不变。

const int ci = 42;
// 顶层const,不能改变ci
int i = ci;
// 当拷贝ci时,会忽略它的顶层const
int *const p = &i;
// 顶层const,不能改变p
*p = 0;
// 可以通过p改变对象的内容是允许的

当实参书初始化形参时会忽略掉顶层const。

同时,当形参有顶层const时,传给它常量对象或者非常量对象都是可以的

void fcn(const int i) {}
// fcn可以读取i,但不能改变i
void fcn(int i) {}
// 这是就存在重复定义,因为函数1会 忽略顶层const,导致函数2与函数1可接收的实参类型可能相同

指针或引用形参与const

可以使用一个非常量初始化一个底层const对象

同时一个普通的引用必须用同类型的对象从初始化

允许常量引用绑定非常量的对象、字面值 、一般表达式

int i = 42;
const int *cp = &i;
// 正确、非常量初始化底层const
const int &r = i;
// 正确、常量引用绑定非常量对象
const int &r2 = 42;
// 正确、常量引用绑定字面值
int  *p = cp;
// 类型不匹配
int &r3 = r;
// 类型不匹配
int &r4 = 42;
// 不能用字面值初始化一个非常量引用
void reset(int &i);
void reset(int *ip);

int i = 0;
const int ci = i;
string::size_type  ctr = 0;

reset(i);
reset(&i);
// 正确

reset(&ci);
reset(ci);
reset(42);
reset(ctr);
// 错误、类型不匹配

应尽量使用常量引用

1、避免误导实参可修改

2、避免限制实参的类型

数组形参

因为数组存在两个性质:

1、不允许拷贝数组

2、使用数组时通常会将其转换成指针

通过性质1可知,无法以值传递的方式使用数组参数

通过性质2可知,当我们为函数传递一个数组时,实际上传递的是指向数组首元素的指针

以下三个函数等价。每个函数的形参类型都是const int*

void print(const int*);
void print(const int[]);
void print(const int[10]); 
// 这里的10表示我们期望数组含有多少元素,实际不一定

当给print函数传递一个数组时,实参自动地转换成指定数组首元素的指针,数组的大小对函数的调用没有影响。

以数组作为形参的函数也必须确保使用数组时不会越界

因为数组是以指针的形式传递给函数的,所以一开始函数并不知道数组的确切尺寸。

管理指针形参有三种常用的技术

1、使用标记指定数组的长度

void print(const char *cp) {
    if (cp)
        while (*cp)
            cout << *cp++;
}
// 处理c风格字符串时会遇到结束标记

使用标准库规范

void print(const int *beg, const int *end) {
    while (beg != end)
        cout << *beg++ << endl;
}
// 使用标准库
// begin()返回首元素指针
// end()返回尾后元素指针

显式传递一个表示数组大小的形参

void print(const int ia[], size_t size) {
    for (size_t i = 0; i < size; i++) 
        cout << ia[i] << endl;
}
// print(ia, end(ia) - begin(ia));

只要传递给函数的size值不超过数组实际的大小,函数就是安全的

数组形参与const

当函数不需要对数组元素执行写操作时,数组形参应该是指向const的指针,当函数确实要改变元素值的时候,才把形参定义成指向非常量的指针

数组引用形参

形参可以是数组的引用。

void print(int (&arr)[10]) {
    for (auto elem : arr)
        cout << elem << endl;
}
// 形参是数组的引用,维度是类型的一部分

f(int &arr[10]);
// 将arr声明成了引用的数组
f(int (&arr)[10]);
// arr是具有10个整数的整型数组的引用

由于数组的大小是构成数组类型的一部分,这样的引用数组无疑限制了函数的可用性。

传递多维数组

多维数组实际上是数组的数组

将多维数组传递给函数时,真正传递的是指向数组首元素的指针,因为我们处理的是数组的数组,所以首元素本身就是一个数组,指针就是一个指向数组的指针,数组第二维的大小都是数组类型的一部分,不鞥省略!

void print(int (*matrix)[10], int rowSize);
// matrix指向数组的首元素,该数组的元素由10个整数构成的数组
// matrix为指向含有10个整数的数组的指针

void print(int matrix[][10], int rowSize);
// matrix声明一个二维数组,实际上形参指向含有10个整数的数组的指针

int *matrix[10];
// 10个指针构成的数组
int (*matrix)[10];
// 指向含有10个整数的数组的指针

 含有可变形参的函数

为了编写能处理不同数量实参的函数,C++提供两种主要的方法

1、如果所有实参的类型相同,可以传递一个名为initializer_list的标准类型

2、如果实参的类型不同,我们可以编写一种特殊的函数,也就是 可变参数模板。

initializer_list形参

如果函数的实参未知但是全部实参的类型都相同,我们可以使用initializer_list类型的形参。用来表示某种特定 类型的值的数组。该类型是一个模板类。大部分操作与vector相同,但是不同之处在于initializer_list对象中的元素永远都是常量值。无法改变对象中元素的值

void error_meg(initializer_list<string> il) {
    for (auto beg = il.begin(); beg != il.end(); ++beg)
        cout << *beg << " ";
    cout << endl;
}
int list_elem(initializer_list<int> il)
{
    int sum = 0;
    for (auto it = il.begin(); it != il.end(); it++)
        sum += *it;
    return sum;
}
int main(int argc, char *argv[])
{
    cout << list_elem({ 1,2,3,4,5,6 }) << endl;
    return 0;
}
// output
21

省略符形参

省略符形参是为了便于C++程序访问某些特殊的C代码而设置的。类似于

void foo(parm_list, ...);
void foo(...);

第一种形式指定了foo函数的部分形参类型,对于这些形参的实参将会执行正常的类型检测,省略符形参所对应的实参无需类型检查

返回类型和return语句

return语句终止当前正在执行的函数并将控制权返回到调用该函数的地方。

无返回值函数

没有返回值的return语句只能用在返回类型是void的函数中,返回void的函数如果没有return语句,则函数的最后一句后会隐式执行return。

一个返回类型是void的函数也能使用return+表达式语句,不过此时表达式必须是一个返回void的函数。

void printStr()
{
    cout << "It's ok!!!" << endl;
}

void print()
{
    return printStr();
}
int main(int argc, char *argv[])
{
    print();
    return 0;
}
// output 
It's ok!!!

有返回值函数

只要函数的返回类型不是void,则该函数内每条return必须返回一个值。返回值的类型必须与函数的返回类型相同,或者能隐式转换成函数的返回类型

值是如何被返回的

返回的值用于初始化调用点的一个临时量,该临时量就是函数调用的结果。

如果函数返回一个值,返回值被拷贝到调用点、

如果函数返回一个引用,该引用仅是它所引对象的一个别名。

不要返回局部变量的指针或引用

函数完成后,它所占用的存储 空间也随之被释放掉。因此,函数终止意味着函数变量的引用将指向不再有效的区域。

如果我们想要确保返回值安全,我们需要引用哪些在函数之前就已经存在的对象

返回类类型的函数和调用运算符

调用运算符()与点运算符.和箭头运算符->相同。并且符合左结合律。

引用返回左值 

函数的返回类型决定函数调用是否是左值。调用一个返回引用的函数得到左值,返回其它类型得到右值。

我们能为返回类型是非常量引用的函数的结果赋值

char& get_val(string &str, string::size_type ix)
{
    return str[ix];
}

int main(int argc, char *argv[])
{
    string s("a value");
    cout << s << endl;
    get_val(s, 0) = 'A';
    cout << s << endl;
    return 0;
}
// Output
a value
A value

 返回数组指针

如果想定义一个返回数组的指针,可以使用类型别名来重写数组名。

typedef int arrT[10];
using arrT = int[10];
arrT* func(int i);
// func返回一个指向含有10个整数的数组的指针

声明一个返回数组指针的函数

如果想在声明func时不使用类型别名,那么就要牢记被定义的名字后面数组的维度

int arr[10];
// arr是一个含有10个整数的数组
int *p1[10];
// p1是含有10个指针的数组
int (*p2)[10];
// p2是一个指针,指向含有10个整数的数组

返回数组指针的函数形式如下

Type (*function(parameter_list)) [dimension]

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

func(int i)表示调用func函数时需要一个int类型的实参

(*func(int i))意味着我们可以对函数调用的结果执行解引用操作

(*func(int i))[10]表示解引用func的调用将得到一个大小是10的数组。

int (*func(int i))[10]表示数组中的元素是int类型

尾置返回类型

尾置返回类型跟在形参列表后面并以一个->符号开头。为了表示函数真正的返回类型跟在形参列表之后 ,我们在本该出现返回类型的地方放置一个auto

auto func(int i)->int(*)[10];

func函数返回一个指针,该指针指向一个含有10个整数的数组

使用decltype

如果我们知道函数返回的指针将指向哪个数组,就可以使用decltype关键字声明返回类型。

例如,下面的函数返回一个指针,该指针根据参数i的不同指向两个已知数组的某一个

int odd[] = {1,3,5,7,9};
int even[] = {0,2,4,6,8};

decltype(odd) *arrPtr(int i)
{
    return (i % 2) ? &odd : &even;
}

decltype表示arrPtr返回类型是个指针,并且该指针所指的对象与odd类型一致,因为odd是数组,所以arrPtr返回一个指向含有5个整数的数组的指针。

decltype并不负责将数组类型转换成对应的指针,所以decltype结果是一个数组,要想表示arrPtr返回指针还必须在函数声明时加一个*符号。

// 声明一个函数,使其返回数组的引用并且该数组包含10个string对象
string (&func(string (&arrStr)[10]))[10];
// 类型别名
using arrS = string[10];
arrS& func1(arrS& arr);

// 尾置返回类型
auto func2(arrS& arr)->string(&)[10];

// decltype
string arrT[10];
decltype(arrT)& func3(arrS& arr);
// 修改arrPtr使其返回数组的引用
auto arrRef(int i)->int(&)[10];
// 实践
int odd[] = { 1,3,5,7,9 };
int even[] = { 0,2,4,6,8 };

decltype(odd) *arrPtr(int i)
{
    return (i % 2) ? &odd : &even;
}

auto arrRef(int i)->int(&)[5]
{
    return (i % 2) ? odd : even;
}

int main()
{
    int (*x)[5] = arrPtr(0);
    int (&y)[5] = arrRef(1);
    for (int i = 0; i != 5; i++)
        cout << (*x)[i] << " ";
    cout << endl;
    for (int i = 0; i != 5; i++)
        cout << y[i] << " ";
    cout << endl;
    return 0;
}
0 2 4 6 8
1 3 5 7 9
请按任意键继续. . .

 

posted @ 2017-12-20 22:09  immjc  阅读(318)  评论(0编辑  收藏  举报