练习 6.31 什么情况下返回的引用无效?什么情况下返回常量的引用无效?
当返回的是局部变量的引用时引用无效,当我们希望返回的对象被修改时,返回常量的引用无效。
练习 6.32 下面的函数合法吗?如果合法,说明其功能;如果不合法,修改其中的错误并解释原因。
int &get(int *array, int index) { return array[index]; }
int main()
{
int ia[10];
for (int i = 0; i != 10; ++i)
get(ia, i) = i;
}
函数get是合法的,其作用是根据索引取得数组中元素的引用。
练习 6.33 编写一个递归函数,输出vector
对象的内容。
v#include<iostream>
#include<vector>
using namespace std;
void print_vec(vector<int> &vec, vector<int>::iterator p) {
if (p != vec.end())
{
cout << *p << " ";
print_vec(vec, p + 1);
}
else {
cout << endl;
return;
}
}
int main(int argc, char **argv)
{
vector<int> ve{ 1,2,3,4,5,6,7,8 };
print_vec(ve, ve.begin());
return 0;
}
练习 6.34 如果factorial
函数的停止条件如下所示,将发生什么?
if (val != 0)
如果val为正数,从结果上来说没有区别(多乘了个 1);
如果val为负数,那么递归永远不会结束。
练习 6.35 在调用factorial 函数时,为什么我们传入的值是 val-1 而非 val–?
因为val–将传入val原来的值,这样会导致递归进入死循环。
练习 6.36 编写一个函数声明,使其返回数组的引用并且该数组包含10个string
对象。不用使用尾置返回类型、decltype
或者类型别名。
string (&func())[10];
练习 6.37 为上一题的函数再写三个声明,一个使用类型别名,另一个使用尾置返回类型,最后一个使用decltype
关键字。你觉得哪种形式最好?为什么?
using sarr = string(&)[10];
sarr func();
typedef string sarr[10];
sarr &func();
auto func()->string(&)[10];
string s[10];
decltype(s) &func();
我觉得尾置返回类型最好。
练习 6.38 修改arrPtr
函数,使其返回数组的引用。
decltype(odd)& arrPtr(int i)
{
return (i % 2) ? odd : even;
}
练习 6.39 说明在下面的每组声明中第二条语句是何含义。如果有非法的声明,请指出来。
(a) int calc(int, int);
int calc(const int, const int);
(b) int get();
double get();
(c) int *reset(int *);
double *reset(double *);
- a)非法,拥有顶层
const
和没有拥有顶层const
的函数无法区分开来。 - b)非法,函数重载应该在形参的数量和类型上区分开来。
- c) 合法
练习 6.40 下面的哪个声明是错误的?为什么?
(a) int ff(int a, int b = 0, int c = 0);
(b) char *init(int ht = 24, int wd, char bckgrnd);
- b) 因为若函数的某个形参被赋予了默认值,则其后面的所有形参都应有默认值。