C++ 在模板函数中强制进行二次名称查找

标签 c++ templates name-lookup

有什么方法可以强制 C++ 编译器在模板实例化期间(而不是之前)对给定符号执行名称查找?

给定以下代码:

template <class T>
auto wrapper( T t ) -> decltype( f( t ) )
{
    return f( t );
}

unsigned char f( int x ) { return x % 256; }
unsigned char f( unsigned char x ) { return x; }

int main( int, char ** )
{
    auto x = wrapper( 3100 );
    return 0;
}

有什么我可以做的(除了将 f 的定义移动到顶部)以使该代码编译并给出与 f< 的所有定义相同的结果wrapper 定义之前可用吗?

我找不到任何东西,可能是因为我不知道如何正确表达这个问题。如果有帮助,f 的所有参数类型都可以假定为用户定义的类型。

最佳答案

Is there some way to force C++ compilers to perform name lookup for a given symbol during template instantiation (and not before)?

是的。首先,名称必须是依赖的。 wrapper 中的名称 f 当用作 f(t) 时是依赖的,因为 t 是类型依赖的。 [临时部门]/1:

In an expression of the form:

        postfix-expression ( expression-list opt )

where the postfix-expression is an unqualified-id, the unqualified-id denotes a dependent name if

  • any of the expressions in the expression-list is a pack expansion (14.5.3),
  • any of the expressions in the expression-list is a type-dependent expression (14.6.2.2), or
  • if the unqualified-id is a template-id in which any of the template arguments depends on a template parameter.

问题是在模板本身之后声明的名称,即仅在实例化而非定义上下文中,只能使用参数相关名称查找 找到。您的 f 重载仅采用基本类型,但根据 [basic.lookup.argdep]/2,那些没有与之关联的全局命名空间:

If T is a fundamental type, its associated sets of namespaces and classes are both empty.

因此,如果参数与参数的类型相同,则永远找不到您声明的 f。一个小技巧可以提供帮助:

template <typename T>
struct refwrap
{
    T&& t;
    refwrap(T&& t) : t(std::forward<T>(t)) {}
    operator T&&() {return std::forward<T>(t);}
};

template <typename T>
auto make_refwrap( T&& t ) -> refwrap<T> // making use of reference collapsing
{ return {std::forward<T>(t)}; }         // inside refwrap to get forwarding

此模板在全局命名空间中声明时,将导致 ADL 考虑它。重写wrapper如下:

template <class T>
auto wrapper( T t ) -> decltype( f( make_refwrap(t) ) )
{
    return f( make_refwrap(t) );
}

Demo .不过,这不是正确的做法,因为在更复杂的情况下它会失败。

关于C++ 在模板函数中强制进行二次名称查找,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28187170/

相关文章:

c++ - 如何在与 C++ 中的派生类型匹配的元素上创建迭代器?

c++ - 重复的可变参数模板参数

go - 在不知道你是否在那个包中的情况下引用 Go 包中的符号?

c++ - 什么是 "Argument-Dependent Lookup"(又名 ADL,或 "Koenig Lookup")?

c++ - 通过内部函数或指令手动控制 Intel MIC SIMD 操作

c++构造函数发生故障

c++ -/usr/bin/ld : cannot find -lcassandra

html - 固定表格中的文本位置

c++ - 根据参数类型获取匹配的重载

C++从具有多重继承模糊函数调用的相同基模板类覆盖函数