C++: Где мое значение?

Пожалуйста, рассмотрите следующий код:

#include <iostream>

template <class T>
class value_wrapper
{
public:

    value_wrapper(T& pv) : v(pv) { std::cout<< "CONS.REF:" << pv << " AT:" << (void*)this << std::endl; }
    value_wrapper(T&& pv) : v(pv) { std::cout<< "CONS.UNIREF:" << pv << " AT:" << (void*)this << std::endl; }
    virtual ~value_wrapper() { std::cout<< "DEST:" << v << " AT:" << (void*)this << std::endl; }
    value_wrapper(const value_wrapper& ov) : v(ov.v) { std::cout<< "CONS.COPY.REF:" << v << " AT:" << (void*)this << std::endl; }
    value_wrapper(value_wrapper&& ov) : v(ov.v) { std::cout<< "CONS.COPY.UNIREF:" << v << " AT:" << (void*)this << std::endl; }
    value_wrapper<T>& operator = (value_wrapper<T>&& ov)
    {
        std::cout<< "ASSI.UNIREF: OF " << v << " AT:" << (void*)this << " TO:" << ov.v << " AT:" <<(void*)&ov << " ADR.VAL:" << (void*)(&ov.v)<< std::endl;
        v = ov.v;
        return *this;
    }

private:
    template <typename V> friend value_wrapper<V> operator - (const value_wrapper<V> v1, const V& v2);
    T& v;
};

template<typename T>
value_wrapper<T> operator - (const value_wrapper<T> v1, const T& v2)
{
    T f = v1.v - v2;
    value_wrapper<T> res(f);
    std::cout << "MINUS: RESULT:" << res.v << " AT:" << (void*)&res << " ADR.VAL:" << &res.v <<  std::endl;
    return res;
}


template<typename X>
value_wrapper<X> _ (X& a)
{
    return value_wrapper<X>(a);
}

int main()
{
    int a = 5;

    std::cout << "BEFOR:" << a<< std::endl;

    _(a) = _(a) - 1;

    std::cout << "AFTER:" << a<< std::endl;

    return 0;

}

И их онлайн присутствие:

(Плохой) http://cpp.sh/7yav

(Хороший) http://coliru.stacked-crooked.com/a/ea7363eaba68a336

В то время как первый выводит:

BEFOR:5
CONS.REF:5 AT:0x761cebd52310
CONS.REF:4 AT:0x761cebd52320
MINUS: RESULT:4 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc
CONS.REF:5 AT:0x761cebd52300
ASSI.UNIREF: OF 5 AT:0x761cebd52300 TO:0 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc
DEST:0 AT:0x761cebd52300
DEST:27644 AT:0x761cebd52320
DEST:0 AT:0x761cebd52310
AFTER:0

а второй выводит:

g++ -std=c++14 -O2 -Wall -pedantic -pthread main.cpp && ./a.out
BEFOR:5
CONS.REF:5 AT:0x7fffc51a5710
CONS.REF:4 AT:0x7fffc51a5720
MINUS: RESULT:4 AT:0x7fffc51a5720 ADR.VAL:0x7fffc51a56fc
CONS.REF:5 AT:0x7fffc51a5700
ASSI.UNIREF: OF 5 AT:0x7fffc51a5700 TO:4 AT:0x7fffc51a5720 ADR.VAL:0x7fffc51a56fc
DEST:4 AT:0x7fffc51a5700
DEST:4 AT:0x7fffc51a5720
DEST:4 AT:0x7fffc51a5710
AFTER:4

Итак, возникает вопрос:

что произошло с значением с того момента, как оно имело правильное значение:

МИНУС: РЕЗУЛЬТАТ: 4 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc

Пока правильное значение не исчезло с того же адреса:

ASSI.UNIREF: OF 5 AT: 0x761cebd52300 TO: 0 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc

Кажется, что я использую тот же компилятор (g++ (Ubuntu 4.8.4-2ubuntu1~14.04) 4.8.4), что и из cpp.sh, потому что я получаю те же результаты...

Это ошибка компилятора или просто мое недопонимание того, как должно работать стандартное значение?

РЕДАКТИРОВАТЬ:

Я ожидал линии:

ASSI.UNIREF: OF 5 AT: 0x761cebd52300 TO: 0 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc

быть:

ASSI.UNIREF: OF 5 AT: 0x761cebd52300 TO:4 AT:0x761cebd52320 ADR.VAL:0x761cebd522cc

то есть: значение 0 должно быть 4 так же, как в другой распечатке, потому что это должно быть правильное значение, а не 0.

1 ответ

Решение

Я думаю, что у вас есть неопределенное поведение здесь:

template<typename T>
value_wrapper<T> operator - (const value_wrapper<T> v1, const T& v2)
{
    T f = v1.v - v2;
    value_wrapper<T> res(f);
    return res;
}

Увидеть? Вы возвращаете value_wrapper<T> который инкапсулирует ссылку на Tи эта ссылка относится к f, Buf f является локальной переменной, поэтому вы возвращаете висячую ссылку, и это UB.

Другие вопросы по тегам