Упражнение 6.31. Когда допустимо возвращение ссылки? Когда ссылки на константу?
Упражнение 6.32. Укажите, корректна ли следующая функция. Если да, то объясните, что она делает; в противном случае исправьте ошибки, а затем объясните все.
int &get(int *arry, int index) { return arry[index]; }
int main() {
int ia[10];
for (int i = 0; i != 10; ++i)
get(ia, i) = i;
}
Упражнение 6.33. Напишите рекурсивную функцию, выводящую содержимое вектора.
Упражнение 6.34. Что случится, если условие остановки функции factorial()
будет таким:
if (val != 0)
Упражнение 6.35. Почему в вызове функции factorial()
мы передали val-1
, а не val--
?
6.3.3. Возвращение указателя на массив
Поскольку копировать массив нельзя, функция не может возвратить его. Но функция может возвратить указатель или ссылку на массив (см. раздел 3.5.1). К сожалению, синтаксис, обычно используемый для определения функций, которые возвращают указатели или ссылки на массив, довольно сложен. К счастью, такие объявления можно упростить. Например, можно использовать псевдоним типа (см. раздел 2.5.1):
typedef int arrT[10]; //
//
using arrtT = int[10]; //
//
arrT* func(int i); //
//
где arrT
— это синоним для массива из десяти целых чисел. Поскольку нельзя возвратить массив, мы определяем тип возвращаемого значения как указатель на этот тип. Таким образом, функция func()
получает один аргумент типа int
и возвращает указатель на массив из десяти целых чисел.
Чтобы объявить функцию func()
, не используя псевдоним типа, следует вспомнить, что размерность массива следует за определяемым именем:
int arr[10]; //
int *p1[10]; //
int (*p2)[10] = &arr //
Подобно этим объявлениям, если необходимо определить функцию, которая возвращает указатель на массив, размерность должна следовать за именем функции. Однако функция имеет список параметров, который также следует за именем. Список параметров предшествует размерности. Следовательно, функция, которая возвращает указатель на массив, имеет такую форму:
Как и в любом другом объявлении массива,
— это тип элементов, а
— это размер массива. Круглые скобки вокруг части (*
необходимы по той же причине, по которой они были нужны при определили указателя p2
. Без них мы определили бы функцию, которая возвращает массив указателей.
В качестве конкретного примера рассмотрим следующее объявление функции func()
, не использующей псевдоним типа:
int (*func(int i))[10];
Чтобы понять это объявление, имеет смысл прочитать его следующим образом:
• func(int)
указывает, что функцию func()
можно вызвать с аргументом типа int
;
• (*func(int))
указывает, что можно обратиться к значению результата этого вызова;
• (*func(int))[10]
указывает, что обращение к значению результата вызова функции func()
возвращает массив из десяти элементов;
• int (*func(int))[10]
указывает, что типом элементов этого массива является int
.
func()
— с использованием ->
. Чтобы сообщить о том, что возвращаемое значение следует за списком параметров, ключевое слово auto
располагается там, где обычно присутствует тип возвращаемого значения:
//
// из десяти целых чисел
auto func(int i) -> int(*)[10];