Указатели и ссылки

Кувшинов Д.Р.

2016


Общее оглавление


Ссылки

Ссылка reference — механизм языка программирования (C++), позволяющий привязать имя к значению. В частности, ссылка позволяет дать дополнительное имя переменной и передавать в функции сами переменные, а не значения переменных.

Синтаксически ссылка оформляется добавлением знака & (амперсанд) после имени типа. Ссылка на ссылку невозможна.

Ссылка требует инициализации. В момент инициализации происходит привязка ссылки к тому, что указано справа от =. После инициализации ссылку нельзя “отвязать” или “перепривязать”.

Любые действия со ссылкой трактуются компилятором как действия, которые будут выполняться над объектом, к которому эта ссылка привязана. Следующий пример демонстрирует ссылку в качестве дополнительного имени переменной.

int n = 0;
int &r = n; /* теперь r -- ссылка на n
  или второе имя переменной n */
n = 10;
cout << r << '\n'; // выведет 10
r = 20;
cout << n << '\n'; // выведет 20
cout << (&n == &r) << '\n'; // выведет 1, т.е. истина

Казалось бы, зачем нам второе имя переменной? Ответа может быть, по крайней мере, два.

  1. Что-то имеет слишком длинное, неудобное название. Привязав к нему ссылку, мы получим более удобное, короткое локальное название. При этом мы можем не указывать тип этого “чего-то”, можно использовать вместо типа ключевое слово auto:
auto &short_name = some_namespace::some_long_long_name;
  1. Выбор объекта привязки ссылки может происходить во время исполнения программы и зависеть от некоего условия. Пример:
int a = 0, b = 0;
cin >> a >> b;
int &max = a < b? b: a; // привязать к b, если a < b, иначе -- к a
max = 42;
cout << "a = " << a << "; b = " << b << '\n';


Впрочем, основным применением ссылок является передача параметров в функции “по ссылке” и возвращение функциями ссылок на некие внешние объекты.

Передача по ссылке by reference напоминает передачу “по имени”. Таким образом, можно сказать, что, используя ссылки, мы передаём не значения, а сами переменные, содержащие эти значения. В реальности “за ширмой” происходит передача адресов этих переменных. Передача ссылки на переменную, время жизни которой заканчивается, например, возврат из функции ссылки на локальную переменную, приводит к неопределённому поведению.

Ранний пример использования ссылок для возврата из функции более одного значения представлен в самостоятельной работе 3.

Приведём здесь ещё один пример: функцию, которая возвращает одну из двух переменных, содержащую максимальное значение. Для этого модифицируем предыдущий пример:

int& max_byref(int &a, int &b)
{
  return a < b? b: a;
}

int main()
{
  int x = 0, y = 0; // собственно имена переменных не обязаны совпадать
  cin >> x >> y;
  max_byref(x, y) = 42;
  cout << "x = " << x << "; y = " << y << '\n';
  return 0;
}


Так как при передаче ссылки реально копируется лишь адрес значения, а не само значение, то передав ссылку можно избежать копирования значения. Поэтому ссылки широко используются для передачи в функцию аргументов, которые или запрещено копировать или вычислительно дорого копировать. Типичный пример — объекты string. При копировании строки происходит выделение динамической памяти, копирование всех символов, затем — при удалении этой копии — освобождение памяти. Часто нет никакой необходимости в копировании. Например, следующей функции, считающей количество повторений заданного символа в строке нет нужды копировать строку — можно обойтись ссылкой:

size_t char_freq(const string &s, char c)
{
  size_t freq = 0;
  for (size_t i = 0, sz = s.size(); i != sz; ++i)
    freq += s[i] == c;
  return freq;
}

Обратите внимание на ключевое слово const. Данное ключевое слово позволяет нам указать, что мы хотим ссылку на константу, т.е. функция char_freq использует s как константу и не пытается её изменять, а ссылка нужна для того, чтобы избежать копирования. Рекомендуется использовать const везде, где достаточно константы. Компилятор проверит, действительно ли мы соблюдаем константность.

Ставить слово const можно перед именем типа и после имени типа, это эквивалентные записи.

int x;
const int &r1 = x; // ссылка на x "только для чтения"
int const &r2 = x; // тоже ссылка на x "только для чтения"
int & const r3 = x; // ошибка компиляции, нельзя ставить const после &


Указатели

Общие сведения

Что такое указатель pointer уже рассказывалось во введении.

В C и C++ указатель определяется с помощью символа * после типа данных, на которые этот указатель будет указывать.

Указатель — старший родственник ссылки. Указатели активно использовались ещё в машинных языках и оттуда были перенесены в C. Ссылки же доступны только в C++.

Указатели — простые переменные. Указатели не “делают вид”, что они — те значения в памяти, к которым они привязаны. Чтобы получить указатель на переменную, нужно явно взять её адрес с помощью оператора &. Чтобы обратиться к переменной, на которую указывает указатель, требуется явно разыменовать его с помощью оператора *.

int n = 0;
int *r = &n; // теперь r -- указатель на n
n = 10;
cout << *r << '\n'; // выведет 10
*r = 20;
cout << n << '\n'; // выведет 20
cout << (&n == r) << '\n'; // выведет 1

Так же, как и в случае ссылок, можно использовать ключевое слово const, чтобы создать указатель на константу.

int x = 0, y = 1;
const int *p1 = &x; // указатель на x "только для чтения"
y = *p1;  // можно
*p1 = 10; // ошибка компиляции: нельзя изменить константу *p1
p1 = &y;  // можно: сам указатель p1 не является константой

int const *p2 = &x; // тоже указатель на x "только для чтения", всё аналогично p1
int * const p3 = &x; // теперь константа -- сам указатель
y = *p3;  // можно
*p3 = 10; // тоже можно!
p3 = &y;  // ошибка компиляции: нельзя изменить константу p3

const int * const p4 = &x; /* комбо:
  теперь у нас константный указатель на x "только для чтения" */
  
y = *p4;  // можно
*p4 = 10; // ошибка компиляции: нельзя изменить константу *p4
p4 = &y;  // ошибка компиляции: нельзя изменить константу p4

Указатели можно сравнивать друг с другом. Указатели равны, если указывают на один и тот же объект, и не равны в противном случае.

Указатели можно передавать в функции и возвращать из функций как и любые “элементарные” значения. Ещё пример с указателями:

int* max_byptr(int *a, int *b)
{
  return *a < *b? b: a;
}

int main()
{
  int x = 0, y = 0; // собственно имена переменных не обязаны совпадать
  cin >> x >> y;
  *max_byref(&x, &y) = 42;
  cout << "x = " << x << "; y = " << y << '\n';
  return 0;
}

Для обращения к полю структуры по указателю на объект структуры предусмотрен специальный оператор -> (“стрелка”).

struct Point { float x, y; };
Point a = { 20, 30 };
cout << a.x << ' ' << a.y << '\n'; // > 20 30
Point *p = &a;
p->x = 42;
(*p).y = 23; // то же самое, что p->y = 23;
cout << a.x << ' ' << a.y << '\n'; // > 42 23


В отличие от ссылок, указатели не обязательно инициализировать. Указатели можно инициализировать специальным значением нулевой указатель nullptr, которое сигнализирует об отсутствии привязки указателя к чему-либо. Присваивание указателю другого адреса меняет его привязку. Это позволяет использовать указатели там, где семантика ссылок слишком сильно ограничивает наши возможности.

Соответственно, ограничения, накладываемые на ссылки по сравнению с указателями, позволяют, с одной стороны, защитить программиста от ряда ошибок, и, с другой стороны, открывают ряд возможностей оптимизации кода для компилятора. Ссылки используются там, где нет нужды в “полноценных” указателях или есть желание не перегружать код взятиями адреса и разыменованиями.

Наличие нулевого указателя позволяет, например, возвращать указатель на искомый объект и в том случае, когда ничего не было найдено. Просто в этой ситуации возвращаем нулевой указатель, а принимающая сторона должна быть готова к такому развитию событий. Указатель автоматически преобразуется к булевскому значению: нулевой указатель даёт false, прочие указатели дают true, поэтому, если p — указатель, то

if (p) ...

есть то же самое, что

if (p != nullptr) ...

И напротив,

if (!p) ...

есть то же самое, что

if (p == nullptr) ...


Например, поиск самого левого нуля в массиве чисел с плавающей точкой может быть записан так:

// Ищет нулевой элемент в диапазоне [from, to).
// Возвращает нулевой указатель, если нуль не был найден.
float* find_next_zero(float *from, float *to)
{
  for (; from != to; ++from)
    if (*from == 0.f)
      return from; // нашли
  return nullptr; // ничего не нашли
}

int main()
{
  float num[] { 1, 2, 3, 0, 3, 4 };
  if (auto zero_pos = find_next_zero(num, num + sizeof(num)/sizeof(num[0])))
    cout << (zero_pos - num) << '\n';
  else
    cout << "zero not found\n"; // невозможно!
  return 0;
}

Данный пример использует арифметику указателей и массивы. Данная тема освещена в разделе массивы и ссылки.


Бестиповый указатель

Вместо типа данных при объявлении указателя можно поставить ключевое слово void. Данное ключевое слово означает, что мы описываем указатель “на что угодно”, т. е. просто адрес в памяти. Любой указатель автоматически приводится к типу void*бестиповому указателю typeless pointer. Прочие указатели, соответственно, называются типизированными или типизованными typed. Приведение от void* к типизованному указателю возможно с помощью оператора явного приведения типа.

В C бестиповые указатели широко применяются для оперирования кусками памяти или реализации обобщённых функций, которые могут работать со значениями разных типов. В последнем случае конкретный тип маскируется с помощью void (“пустышка”). При использовании таких функций обычно приходится где-то явно приводить тип указателей. C++ позволяет отказаться от подобной практики благодаря поддержке полиморфизма и обобщённого программирования (материал 2-го семестра).

#include <iostream>
#include <iomanip>  // setw -- ширина поля вывода, hex -- вывод в 16-ричной системе
#include <cstring>
using namespace std;
// Ещё один способ получить битовое представление числа с плавающей точкой.
int main()
{
  unsigned char buffer[sizeof(float)];
  // Настройка потока вывода.
  cout.fill('0');        // Заполнять нулями.
  cout.setf(ios::right); // Выравнивать по правому краю.
  for (float x; cin >> x; )
  {
    // Скопировать побайтово память x в память buffer.
    memcpy(buffer, &x, sizeof(float));
    // Вывести каждый байт buffer в 16-ричной форме.
    for (int byte: buffer)
      cout << setw(2) << hex << byte << ' ';
    cout << '\n';
  }
}

О цикле for (int byte: buffer) см. здесь.


Указатель на указатель

Так как указатель — обычная переменная, возможен указатель на указатель. И указатель на указатель на указатель. И указатель (на указатель) n раз для натурального n. Максимальный уровень вложенности задаётся компилятором, но на практике уровни больше 2 практически не используются.

int n = 4;
int *p = &n;       // уровень косвенности 1
*p = 5;
cout << n;  // выведет 5
int **pp = &p;     // уровень косвенности 2
**p = 6;
cout << n;  // выведет 6
int ***ppp = &pp;  // уровень косвенности 3
***p = 7;
cout << n;  // выведет 7

Процитируем Three Star Programmer:

Система ранжирования C-программистов.

Чем выше уровень косвенности ваших указателей (т. е. чем больше “*” перед вашими переменными), тем выше ваша репутация. Беззвёздочных C-программистов практически не бывает, так как практически все нетривиальные программы требуют использования указателей. Большинство являются однозвёздочными программистами. В старые времена (ну хорошо, я молод, поэтому это старые времена на мой взгляд) тот, кто случайно сталкивался с кодом, созданный трёхзвёздочным программистом, приходил в благоговейный трепет.

Некоторые даже утверждали, что видели трёхзвёздочный код, в котором указатели на функции применялись более чем на одном уровне косвенности. Как по мне, так эти рассказы столь же правдивы, сколь рассказы об НЛО.

Просто чтобы было ясно: если вас назвали Трёхзвёздочным Программистом, то обычно это не комплимент."

Условия для проверки себя на “трёхзвёздность” перечислены на другой странице того же сайта.

В случае C указатели на указатели (уровень косвенности 2) используются довольно часто, например, для возвращения указателя из функции, которая возвращает ещё что-то, или для организации двумерных массивов. Пример такой функции из Windows API:

DWORD WINAPI GetFullPathName(
  _In_  LPCTSTR lpFileName,
  _In_  DWORD   nBufferLength,
  _Out_ LPTSTR  lpBuffer,
  _Out_ LPTSTR  *lpFilePart
);

Функция принимает имя файла как указатель на си-строку lpFileName, а также размер буфера nBufferLength в символах и адрес буфера lpBuffer, куда записывается в виде си-строки полное имя файла. Функция возвращает длину строки, записанной в буфер, или 0, если произошла ошибка. Кроме того, последний параметр функции — указатель на указатель на си-строку lpFilePart, который используется, чтобы вернуть из функции указатель на последнюю часть имени файла, записанного в буфер.

В случае C++ с помощью ссылок и Стандартной библиотеки можно вообще избежать использования “классических” указателей. Так что “беззвёздочный” C++-программист возможен.

См. также примеры в подразделе, посвящённом указателям на функции.


Неограниченный уровень косвенности

Несмотря на ограниченность применения уровня косвенности выше двух, довольно часто встречается то, что можно назвать неограниченным уровнем косвенности или рекурсивным типом данных. Типичный (и простейший) пример — структура данных, называемая “связанный список” linked list.

Следующий пример демонстрирует использование связанного списка для чтения последовательности строк и вывода этой последовательности в обратном порядке:

struct Line
{
  Line *prev;
  string line;
};

int main()
{
  Line *last = nullptr;

  // Чтение строк.
  for (string line; getline(cin, line);)
  {
    Line *new_line = new Line;
    new_line->prev = last;
    new_line->line = line;
    last = new_line;
  }

  // Вывод строк в обратном порядке.
  while (last)
  {
    cout << last->line << '\n';
    Line *old_line = last;
    last = last->prev;
    delete old_line;
  }
  
  return EXIT_SUCCESS;
}

C++, HTML

Упражнение. Попробуйте изменить этот пример так, чтобы введённые строки выводились в том же порядке, в котором были введены.


Указатели на функции

Язык C позволяет определять указатели на функции (в указателе хранится адрес точки входа в функцию) и вызывать функции по указателю. Таким образом, можно во время исполнения программы выбирать какая именно функция будет вызвана в конкретной точке, выбирая значение указателя. Язык C++ позволяет создавать также и ссылки на функции, но ввиду того, что ссылка после инициализации не может быть изменена, область применения ссылок на функции весьма узка.

Функцией высшего порядка higher order function называют функцию, принимающую в качестве параметров другие функции. Функции высшего порядка — одно из базовых понятий функционального программирования. Единственная форма функций высшего порядка в C — функции, принимающие указатели на функции. Язык C++ расширяет круг доступных форм функций высшего порядка, но в примерах ниже мы ограничимся возможностями C.

Простой пример использования указателя на функцию — функция, решающая уравнение вида f(x) = 0, где f(x) — произвольная функция. Конкретные функции f можно передавать по указателю. Приведение функций к указателю на функцию и наоборот производится неявно автоматически, поэтому при присваивании указателю адреса конкретной функции можно не использовать оператор взятия адреса &, а при вызове функции по указателю — не использовать оператор разыменования * (поведение, аналогичное поведению с массивами).

/// Тип "правая часть уравнения" -- функция одного действительного параметра.
typedef double (*Unary_real_function)(double);

/// Точность приближённого решения, используемая по умолчанию.
const double Tolerance = 1e-8;

/// Алгоритм численного решения уравнения f(x) = 0 на отрезке [a, b] делением отрезка пополам.
/// Данный алгоритм является вариантом двоичного поиска.
double nsolve(Unary_real_function f, double a, double b, double tol = Tolerance)
{
  using namespace std;
  assert(f != nullptr);
  assert(a < b);
  assert(0. <= tol);
  for (auto fa = f(a), fb = f(b);;)
  {
    // Проверим значения функции на концах отрезка.
    if (fa == 0.)
      return a;
    if (fb == 0.)
      return b;

    // Делим отрезок пополам.
    const auto mid = 0.5 * (a + b); // середина отрезка
    if (mid <= a || b <= mid)
      return abs(fa) < abs(fb)? a: b;
    if (b - a <= tol)
      return mid;

    // Выберем одну из половин в качестве уточнённого отрезка.
    const auto fmid = f(mid);
    if (signbit(fa) != signbit(fmid))
    {
      // Корень на левой половине.
      b = mid;
      fb = fmid;
    }
    else
    {
      assert(signbit(fb) != signbit(fmid));
      // Корень на правой половине.
      a = mid;
      fa = fmid;
    }
  }
}

C++, HTML

(О typedef см. ниже.)


Довольно типичной областью применения указателей на функции является связывание источников (регистраторов) некоторых событий, обычно определяемых в составе некоторой библиотеки, и обработчиков событий, предоставляемых пользователем этой библиотеки. Обработчики событий (функции) вызываются автоматически по переданным указателям. Такие функции также называются функциями обратного вызова callback functions или колбеками callbacks. Например, при щелчке мышью по элементу графического интерфейса вызывается функция-обработчик этого события, “зарегистрированная”, путём передачи её адреса библиотеке графического интерфейса.

В качестве простого примера применения функции обратного вызова рассмотрим функцию, занимающуюся поиском набора корней уравнения f(x) = 0 на заданном отрезке. Сама функция будет работать по достаточно простому алгоритму (который, естественно, не гарантирует, что будут найдены все или даже какие-то из существующих на отрезке корней): предполагаем, что есть некая функция, способная найти один корень на отрезке, если он там есть (например, функция nsolve из примера выше). Теперь берём исходный отрезок поиска [a, b] и некоторое значение “шага” step и проходим по этому отрезку с этим шагом, проверяя участки [a + i step, min(b, a + (i + 1)step], i = 0, … пока не пересечём правую границу отрезка. На каждом участке проверяем, являются ли его границы корнями, и есть ли на нём корень (принимает ли функция f разнознаковые значения на границах). В последнем случае используем “решатель” вроде nsolve (переданный по указателю), чтобы найти корень. Каждый найденный корень — это событие, вызываем для него “обработчик” — функцию обратного вызова по указателю report.

/// Тип "решатель уравнения на отрезке" -- функция вроде nsolve, определённой выше.
typedef double (*Equation_solver)(Unary_real_function, double a, double b, double tol);
/// Тип функции, вызываемой для каждого корня. 
/// Процесс поиска останавливается, если эта функция возвращает ложь.
typedef bool (*Root_reporter)(double);

/// Применяет заданный алгоритм поиска корня на отрезке, 
/// разбивая заданный отрезок [a, b] на отрезки одинаковой длины step (кроме, возможно, последнего).
/// Для каждого найденного корня вызывает функцию report (callback-функция).
/// Возвращает правую границу пройденного участка (идёт слева направо по заданному отрезку).
double repeated_nsolve
  (
    Unary_real_function f, double a, double b,
    double step, // шаг на отрезке
    Root_reporter report,
    double x_tol = TOLERANCE, // чувствительность по аргументу
    double f_tol = TOLERANCE, // чувствительность по значению функции
    Equation_solver solver = nsolve
  )
{
  assert(x_tol >= 0. && f_tol >= 0.);
  assert(a <= b);
  assert(step > 0.);
  assert(f && report && solver);

  using namespace std;
  double left = a, f_left = f(left);
  bool f_left_zero = abs(f_left) <= f_tol;
  
  // Корень на левой границе исходного отрезка?
  if (f_left_zero && !report(left))
    return left;

  while (left != b)
  {
    // Правая граница очередного участка.
    const double right = fmin(b, left + step), f_right = f(right);
    const bool f_right_zero = abs(f_right) <= f_tol;
    
    // Корень на правой границе участка?
    if (f_right_zero && !report(right))
      return right;

    // Есть корень внутри участка?
    if (!(f_left_zero || f_right_zero) && signbit(f_left) != signbit(f_right))
    {
      const double root = solver(f, left, right, x_tol);
      if (!report(root))
        return root;
    }

    // Передвинуть левую границу.
    left = right;
    f_left = f_right;
    f_left_zero = f_right_zero;
  }

  return b;
}

C++, HTML

См. также 0610-global_solve.cpp (развитие) и 2040-nsolve.cpp (аналог).


Следующий пример демонстрирует “двухзвёздное программирование” и использование указателя на функцию для определения порядка сортировки массива строк с помощью стандартной функции qsort.

#include <cstdlib>  // qsort
#include <cstring>  // strcmp
#include <iostream>
using namespace std;

// Функция сравнения строк.
int line_compare(const void *left, const void *right)
{
  // Обращаем словарный порядок, поменяв местами left и right.
  return strcmp(*(const char**)right, *(const char**)left);
}

int main()
{
  const char *lines[]
  {
    "may the force be with you",
    "this is it",
    "so be it",
    "it is a good day to die",
    "through the time and space",
    "the light shines in the darkness"
  };

  // Сортировать: массив, количество элементов
  qsort(lines, sizeof(lines) / sizeof(lines[0]),
    // размер элемента, функция сравнения.
    sizeof(lines[0]), line_compare);

  // Распечатаем результат сортировки.
  for (auto line : lines)
    cout << line << '\n';

  return EXIT_SUCCESS;
}

Функция qsort является частью Стандартной библиотеки C. Стандартная библиотека C++ предлагает более удобную и эффективную функцию sort (определённую в заголовочном файле <algorithm>), однако её рассмотрение выходит за пределы темы данного раздела.

Следующий пример является развитием примера со списком из предыдущего подраздела и использует бестиповые указатели, указатели на указатели и указатели на функции для управления “обобщённым” связанным списком в стиле C. Звенья такого списка могут содержать произвольные данные. Основное требование к звеньям списка — наличие в начале звена указателя на следующее звено, фактически каждый предыдущий указатель указывает на следующий.

/// Возвращает ссылку на указатель на следующее звено звена link.
void*& next(void *link)
{
  return *(void**)link;
}

/// Вставляет link перед head и возвращает link (теперь это -- новая голова списка).
void* insert_head(void *head, void *link)
{
  next(link) = head;
  return link;
}

/// Вычисляет длину списка.
size_t size(void *head)
{
  size_t sz = 0;
  for (; head; head = next(head))
    ++sz;
  return sz;
}

/// Указатель на функцию, выполняющую удаление звена.
using Link_delete = void(*)(void*);

/// Удаляет список, используя пользовательскую функцию удаления.
void delete_list(void *head, Link_delete link_delete)
{
  while (head)
  {
    auto next_head = next(head);
    link_delete(head);
    head = next_head;
  }    
}

Теперь сама программа, выводящая строки в обратном порядке, упрощается:

/// Звено списка -- одна строка.
struct Line
{
  void *prev;
  string line;
};

/// Вывести строку и удалить объект Line.
void print_and_delete(void *ptr)
{
  auto line = (Line*)ptr;
  cout << line->line << '\n';
  delete line;
}

int main()
{
  Line *head = nullptr;

  // Чтение строк.
  for (string line; getline(cin, line);)
  {
    Line *new_line = new Line;
    new_line->line = line;
    head = (Line*)insert_head(head, new_line);
  }

  // Вывод количества строк -- элементов списка.
  cout << "\nLines: " << size(head) << "\n\n";

  // Вывод строк в обратном порядке.
  delete_list(head, print_and_delete);

  cin.clear();
  cin.ignore();

  return EXIT_SUCCESS;
}

C++, HTML

Впрочем, необходимо отметить, что сочетая такие приёмы со средствами C++, выходящими за пределы “чистого” C, вы рискуете нарваться на неопределённое поведение. Низкоуровневые средства требуют особой внимательности, так как компилятор в таких случаях не страхует программиста. В частности, в общем случае нельзя интерпретировать произвольный указатель как void* и наоборот без выполнения приведения типа. А это может произойти неявно, например, в примере выше мы полагаем, что указатель prev, указывающий на объект структуры Line совпадает с указателем на поле prev этого объекта.


Синтаксическая справка

Правило чтения сложных описаний типов

Конструкции, определяющие переменные или вводящие новые типы в языках C и C++, могут порой иметь довольно запутанный вид. Ниже дано правило, помогающее разобраться в смысле сложных конструкций.

  1. Начиная с имени (в случае typedef, в случае using имя находится вне — см. ниже), читать вправо, пока это возможно (до закрывающей круглой скобки или точки с запятой).
  2. Пока невозможно читать вправо, читать влево (убирая скобки).

Некоторые примеры “расшифровки” типов переменных:

// c (влево) константа char (const и char можно поменять местами)
const char c;

// str (влево) указатель на (влево) константу char (или константный массив из char)
const char* str;

// str (влево) константный (влево) указатель на константу char
const char* const str; 

// n (вправо) массив (вправо) из 10 (влево) int
int n[10];

// n (вправо) массив (вправо) из 10 (влево) указателей на (влево) int
int* n[10];

// n (влево) указатель на (вправо) массив из 10 (влево) указателей на int
int* (*n)[10];

// n указатель на массив из 10 (влево) указателей на (вправо) функции, не принимающие аргументов,
// (влево) возвращающие указатели (влево) на константы типа int
const int* (*(*n)[10])();

Разница между typedef и using

Директива typedef объявляет синоним типа. Используется синтаксис определения переменной, к которой добавили ключевое слово typedef, только вместо собственно переменной вводится синоним типа этой как-бы переменной с её именем.

int * p; // переменная: указатель на int
typedef int * pt; // имя pt -- синоним типа "указатель на int"

pt px; // тоже переменная типа "указатель на int"

В С++11 появилась возможность объявлять синонимы типов с помощью using-директивы в стиле инициализации переменных:

using pointer = type*;

Объявление typedef можно превратить в using-директиву, заменив typedef на using, вставив после using имя типа и знак равно и убрав это имя типа из объявления справа.

// то же, что typedef double (*Binary_op)(double, double);
using Binary_op = double (*)(double, double);

Типы, ассоциируемые с массивами

Пусть N — константа времени компиляции и дано определение

float a[N];

Тогда

Типы, ассоциируемые с функциями

Пусть дано объявление

float foo(int, int);

Тогда


Общее оглавление

Кувшинов Д.Р. © 2016