代码之家  ›  专栏  ›  技术社区  ›  JeJo

调用参数中是否允许typename T?

  •  4
  • JeJo  · 技术社区  · 5 年前

    对不起,我无法为这一期制作更好的标题。

    我写了一个代码 this SO post ; 看起来像:

    #include <iostream>
    #include <string>
    #include <type_traits>
    
    template < typename T, typename ...Ts> T f(const T &a, Ts&&... args)
    {
        return a;
    }
    
    template < typename R, typename T, typename... Ts>
    typename std::enable_if<!std::is_same<R, T>::value, R>::type
    f(typename T, Ts&&... args)
    //^^^^^^^^^^ --->HERE
    {
        return f<R>(std::forward<Ts>(args)...);
    }
    
    int main() {
        std::cout << f<int>('a', std::string{ "string" }, 12); 
        return 0; 
    }
    

    你可以看到,我打错了(大概是吧)

    f(typename T, Ts&&... args)
    //^^^^^^^^^^
    

    然而,代码符合 msvc v19.14 什么时候 -O3 -std=c++11 .

    另一方面,GCC和clang提供编译器错误: https://godbolt.org/z/HugROb

    从…起 GCC :

    error : 'template<class R, class T, class ... Ts> typename std::enable_if<(! std::is_same< <template-parameter-1-1>, <template-parameter-1-2> >::value), R>::type f' conflicts with a previous declaration
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^ ~~~~~~~
    note : previous declaration 'T f(T, Ts&& ...)'
    
    template < typename T, typename ...Ts> T f(T a, Ts&&... args)
    ^
    error : expected nested - name - specifier before 'T'
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^
    error : expected '(' before 'T'
    error : expected primary - expression before '&&' token
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^ ~
    warning : variable templates only available with - std = c++14 or -std = gnu++14
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^
    error : expected ';' before '{' token
    {
    ^
    

    从…起 叮当声 :

    <source>:13 : 74 : error : expected a qualified name after 'typename'
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^
    <source> : 13 : 74 : error : expected a qualified name after 'typename'
    <source> : 13 : 74 : error : expected ')'
    <source> : 13 : 64 : note : to match this '('
    typename std::enable_if<!std::is_same<R, T>::value, R>::type f(typename T, Ts&&... args)
    ^
    <source> : 15 : 34 : error : use of undeclared identifier 'args'; did you mean 'abs' ?
    return f<R>(std::forward<Ts>(args)...);
    ^~~~
    abs
    / usr / include / stdlib.h:837 : 12 : note : 'abs' declared here
    extern int abs(int __x) __THROW __attribute__((__const__)) __wur;
    ^
    

    真的允许这样吗?或者它是msvc编译器中的一个bug?

    1 回复  |  直到 5 年前
        1
  •  5
  •   Davis Herring    5 年前

    解析器指南 typename 可以出现 only before 限定名称(即带有 :: ). 在非限定名称之前永远不需要它,因为此类名称必须引用某些已知声明,因此如果它们引用了类型,则可以引用类型。