【问题标题】:C++ dereference happens after implicit conversionC++ 取消引用发生在隐式转换之后
【发布时间】:2018-05-08 09:52:59
【问题描述】:

今天我看到在函数调用中尝试取消引用参数时,隐式转换发生在实际取消引用操作之前(我相信只有在原始类型不支持取消引用的情况下)。

这个观察可以在这里看到:

struct A{};


struct C
{
    C(const A&)
    {

    }
};


C operator*(const C&);

double f(C);


template <typename T>
struct t
{
    static const bool value = sizeof(f(**static_cast<T*>(NULL))) == sizeof(double);
};


int main(int argc, const char * argv[]) {
    t<A>::value;
}

C++ 标准是否明确提到了这种行为?谢谢。

【问题讨论】:

  • This implicit conversion references 列出“当表达式用作操作数时,带有需要 T2 的运算符”。在您的情况下,T1AT2C。您对A 对象执行一元解引用运算符,没有定义这样的运算符,但是C 有一个可以将A 类型转换为的运算符。然后使用一元解引用运算符。
  • 请解释如何在那里看到它,因为该程序没有可观察到的行为。
  • @molbdnilo 如果没有发生转换,代码将给出诊断(没有匹配的 f 重载)
  • @Someprogrammerdude 好的,所以发生在函数调用中的事实没有相关性,对吗?
  • 函数调用是一个红鲱鱼。在**static_cast&lt;T*&gt;(NULL) 中,您基本上拥有A a; *a; 使*a 编译的唯一方法是operator*(C(a))

标签: c++ language-lawyer implicit-conversion dereference


【解决方案1】:

我们来看看这个表达式:

f(**static_cast<A*>(NULL))

从最里面到最外面,static_cast&lt;A*&gt;(NULL) 是一个A*(虽然更喜欢nullptrNULL,并且也更喜欢使用std::declval&lt;A*&gt;() 来获得“A* 类型的东西”而不是旧的空指针转换方法)。

接下来,*(a prvalue of type A*) 为您提供A 类型的左值。这就是指针取消引用的含义,它不可重载。当事情很容易推理时,这很好。

接下来,*(an lvalue of type A)。为了弄清楚这意味着什么,我们transform 调用了函数符号,我们的候选集是:

第一个项目符号没有找到任何东西,没有A::operator*()。第二个项目符号,对operator*() 的非限定查找将找到函数C operator*(const C&amp;);,因为它在范围内。这是一个可行的候选,因为A 可以通过C(A const&amp;) 转换为C。第三个子弹没有可行的候选者。

由于我们只有一个可行的候选者,因此它无疑是最佳可行的候选者 - 所以*(lvalue of type A) 为我们提供了C 类型的prvalue。

具体回答您的问题:

隐式转换发生在实际取消引用操作之前

是的,必须进行转换才能解决取消引用操作。虽然它实际上不是“取消引用”,但它只是一个一元 operator*()

最后,f(prvalue of type C) 调用我们拥有的f,它给了我们double


请注意,在现代 C++ 中,我建议将检查写成更接近于:

template <typename T>
struct t
    : std::is_same<
        decltype(f(*std::declval<T>())), // <== the type we get when we call f
                                         // on a dereferenced T
        double>
{ };

【讨论】:

  • @xskxzr 如果它在评估的上下文中。
  • 感谢您的回答。函数调用让我认为它负责隐式转换,但正如您所指出的,在上下文中找到的 operator*() 是首先需要它的那个。
猜你喜欢
  • 1970-01-01
  • 2021-06-08
  • 2016-10-08
  • 2014-02-24
  • 1970-01-01
  • 1970-01-01
  • 2013-06-05
  • 2011-10-02
  • 1970-01-01
相关资源
最近更新 更多