C++MiddleTechnical

Что такое вывод типа auto в C++ и как он работает с decltype?

auto выводит тип из инициализатора, снимая ссылки и const. decltype возвращает точный задекларированный тип выражения. decltype(auto) комбинирует оба: выводит тип auto-правилами, но сохраняет ссылки как decltype.

Зачем нужны auto и decltype

До C++11 тип переменной приходилось писать явно — даже когда компилятор однозначно знал его из правой части выражения. auto и decltype решают разные задачи: auto выводит тип из инициализатора, отбрасывая cv-квалификаторы и ссылки; decltype возвращает задекларированный тип выражения ровно так, как он написан — вместе с const, & и &&.

Правила вывода auto

auto работает по тем же правилам, что и вычет шаблонного параметра: ссылки и cv-квалификаторы снимаются, если явно не добавить & или const.

const std::vector<int> v{1, 2, 3};

auto a = v[0];         // int        — копия, const снята
auto& b = v[0];        // const int& — ссылка, const сохранена
const auto c = v[0];   // const int  — явно добавили const

// initializer_list — классическая ловушка
auto d = {42};         // std::initializer_list<int>, НЕ int!

Правила decltype

decltype(expr) имеет два режима:

  • Если expr — имя переменной или член структуры без скобок, возвращает задекларированный тип один к одному.
  • Если expr — любое другое выражение (в том числе (x) с лишними скобками), возвращает тип по категории значения: lvalue → T&, xvalue → T&&, prvalue → T.
int x = 0;
decltype(x)   t1 = x;  // int    — имя переменной
decltype((x)) t2 = x;  // int&   — выражение-lvalue

const std::vector<int> v{1, 2, 3};
decltype(v[0]) r = v[0]; // const int& — operator[] возвращает const int&

decltype(auto) — гибридный режим

C++14 добавил decltype(auto): вывести тип как auto, но применить семантику decltype — то есть сохранить ссылки и cv-квалификаторы.

const std::vector<int> v{1, 2, 3};

auto            a = v[0];          // int        (копия)
decltype(auto)  b = v[0];          // const int& (ссылка сохранена)

// Полезно в шаблонных обёртках:
template<typename F, typename... Args>
decltype(auto) call(F&& f, Args&&... args) {
    return std::forward<F>(f)(std::forward<Args>(args)...);
    // без decltype(auto) ссылочный возврат f() превратился бы в копию
}

Trailing return type и auto в функциях

// До C++14: trailing return type
template<typename T, typename U>
auto add(T a, U b) -> decltype(a + b) {
    return a + b;
}

// C++14: auto сам выведет из return
template<typename T, typename U>
auto add14(T a, U b) {
    return a + b; // тип = тип выражения a+b, ссылка снимается
}

// C++14: decltype(auto) сохраняет ссылку если return — lvalue
decltype(auto) getRef(std::vector<int>& v) {
    return v[0]; // int&, а не int
}

Практический пример: кэш с прокси-доступом

#include <unordered_map>
#include <string>
#include <iostream>

struct Cache {
    std::unordered_map<std::string, int> data;

    // decltype(auto) пробрасывает int& из operator[]
    decltype(auto) get(const std::string& key) {
        return data[key];
    }
};

int main() {
    Cache c;
    c.get("hits") = 42;

    auto val = c.get("hits");          // int — копия
    decltype(auto) ref = c.get("hits"); // int& — ссылка на map-элемент

    ref = 100;
    std::cout << val << ' ' << c.data["hits"] << '\n'; // 42 100
}

Подводные камни

  • Dangling reference через auto&. auto& x = foo(); — если foo() возвращает prvalue, жизнь временного объекта продлевается только до конца полного выражения в большинстве контекстов (в range-for — нет, там Clang/GCC предупреждают).
  • Скобки в decltype меняют тип. decltype(x)int; decltype((x))int&. В decltype(auto) f() { return (x); } функция вернёт ссылку на локальную переменную — UB.
  • auto и std::initializer_list. auto x = {1, 2, 3}; выводит std::initializer_list<int>, а не массив. Часто неожиданно при рефакторинге.
  • auto снимает const с указателя на const. const int* p = ...; auto q = p;q будет const int* (const данных сохранён), но const int* const p = ...; auto q = p; — верхний const снимается, q снова const int*.
  • Прокси-объекты. auto x = vec_of_bool[0]; даёт не bool, а std::vector<bool>::reference — прокси, который ссылается на внутренний буфер. После перераспределения вектора — dangling.
  • decltype(auto) в цепочке возвратов. В глубоко вложенных шаблонах decltype(auto) может неожиданно вернуть T&& на временный объект из вызываемого слоя — UB при использовании результата.
  • Нечитаемый код при злоупотреблении. auto в API-сигнатурах без trailing return type затрудняет понимание публичного контракта; IDE и doxygen показывают auto вместо реального типа.
  • Разное поведение MSVC / GCC / Clang до C++17. В некоторых случаях вывод типа в trailing return type с рекурсивными шаблонами давал разные результаты между компиляторами до стандартизации в C++17.

Common mistakes

  • Объяснять auto и decltype только по синтаксису, без жизненного цикла и стоимости.
  • Игнорировать ошибки, null/empty состояния, порядок инициализации или режим сборки.
  • Давать пример, который работает в демо, но ломается при изменении владельца ресурса.
  • Показывать сырой указатель без объяснения владельца и момента освобождения.

What the interviewer is testing

  • Кандидат формулирует точную модель для auto и decltype, а не только определение.
  • Пример компилируем, безопасен по lifetime и соответствует версии технологии.
  • Названы trade-off, ограничения и диагностируемые симптомы ошибки.

Sources

Related topics