c++ - Почему не требуется использовать typename для зависимых типов в следующем случае?




c++11 templates (2)

Я читал об удалении ссылки типа here .

Это дает следующий пример:

#include <iostream> // std::cout
#include <type_traits> // std::is_same

template<class T1, class T2>
void print_is_same() {
  std::cout << std::is_same<T1, T2>() << '\n';
}

int main() {
  std::cout << std::boolalpha;

  print_is_same<int, int>();
  print_is_same<int, int &>();
  print_is_same<int, int &&>();

  print_is_same<int, std::remove_reference<int>::type>(); // Why not typename std::remove_reference<int>::type ?
  print_is_same<int, std::remove_reference<int &>::type>();// Why not typename std::remove_reference<int &>::type ?
  print_is_same<int, std::remove_reference<int &&>::type>();// Why not typename std::remove_reference<int &&>::type ?
}

type s в признаках std::remove_reference является зависимым типом.

Возможная реализация

template< class T > struct remove_reference      {typedef T type;};
template< class T > struct remove_reference<T&>  {typedef T type;};
template< class T > struct remove_reference<T&&> {typedef T type;};

Но почему он не использует typename std::remove_reference</*TYPE*/>::type ?


type s в признаках std::remove_reference является зависимым типом.

Нет, они не являются зависимыми именами здесь. Аргументы шаблона были явно указаны как int , int& и int&& . Следовательно, типы известны на данный момент.

С другой стороны, если вы используете std::remove_reference с параметром шаблона, например

template <typename T>
void foo() {
    print_is_same<int, typename std::remove_reference<T>::type>();
}

затем вы должны использовать typename чтобы сказать, что std::remove_reference<T>::type является типом, поскольку ваше выражение теперь зависит от параметра шаблона T


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







dependent-name