Mój oryginalny przykład jest złożony.
Dlatego zamieszczam tutaj prostszy przykład i dostarczam odpowiedni paragraf ISO C++ standard.
To prostsze przykładem jest również dostępny na coliru.stacked-crooked.com/
#include <iostream>
struct A
{
A(int i) { std::cout<<"Cstr "<< i<<'\n'; p = new int(i); }
~A() { std::cout<<"Dstr "<<*p<<'\n'; delete p; }
const A& thiz() const { return *this; }
int *p;
};
const A& constref(const A& a)
{
return a;
}
int main()
{
const A& a4 = A(4);
const A& a5 = A(5).thiz();
const A& a6 = constref(A(6));
std::cout << "a4 = "<< *a4.p <<'\n';
std::cout << "a5 = "<< *a5.p <<'\n';
std::cout << "a6 = "<< *a6.p <<'\n';
}
Wyjście z użyciem wiersza poleceń g++-4.8 -std=c++11 -O2 -Wall -pedantic -pthread main.cpp && ./a.out
:
Cstr 4
Cstr 5
Dstr 5
Cstr 6
Dstr 6
a4 = 4
a5 = 0
a6 = 0
Dstr 4
Jak widać, tymczasowe obiekty odwołuje a5
i a6
są niszczone u koniec funkcji odpowiednio thiz
i constref
.
Jest to wyciąg z §12.2 obiektów tymczasowych, gdzie pogrubione części stosuje się w tym przypadku:
Drugi kontekst jest, gdy odwołanie jest związany z tymczasowym. czasowy, do których odniesienia jest związany lub tymczasowy że pełna przedmiotem podobiektu do którego odniesienie wiąże utrzymuje się przez cały okres odniesienia, z wyjątkiem:
- tymczasowa związana z elementem referencyjnym w konstruktorze inicjator ctor (12.6.2) działa dopóki konstruktor nie zakończy pracy.
- Czasowe powiązanie z parametrem referencyjnym w wywołaniu funkcji (5.2.2) trwa do zakończenia pełnego wyrażenia zawierającego połączenie.
- Żywotność tymczasowego powiązania z wartością zwróconą w instrukcja zwrotu funkcji (6.6.3) nie jest przedłużona; tymczasowy jest niszczony na końcu pełnego wyrażenia w instrukcji return.
- tymczasowa związana z wzorcem w nowej-inicjator (5.3.4) utrzymuje do wykonania pełnej ekspresji zawierającego nową-inicjator.
To jest bardziej kompletny przykład:
#include <iostream>
struct A
{
A() { std::cout<<"Cstr 9\n"; p = new int(v = 9); }
A(int i) { std::cout<<"Cstr "<<i<<'\n'; p = new int(v = i); }
A(const A&o){ std::cout<<"Copy "<<o.v<<'\n'; p = new int(v = 10+o.v); }
~A() { std::cout<<"Del "<<v<<' '<<*p<<'\n'; *p = 88; delete p; }
const A& thiz() const { return *this; }
int *p;
int v;
};
const A& constref(const A& a)
{
return a;
}
std::ostream& operator<<(std::ostream& os, const A& a)
{
os <<"{ *p="<< *a.p <<" , v="<< a.v <<" }\n";
return os;
}
int main()
{
std::cout << "---const A a1 = A(1)" "\n";
const A a1 = A(1);
std::cout << "---const A a2 = A(2).thiz()" "\n";
const A a2 = A(2).thiz();
std::cout << "---const A a3 = constref(A(3))" "\n";
const A a3 = constref(A(3));
std::cout << "---const A& a4 = A(4)" "\n";
const A& a4 = A(4);
std::cout << "---const A& a5 = A(5).thiz()" "\n";
const A& a5 = A(5).thiz();
std::cout << "---const A& a6 = constref(A(6))" "\n";
const A& a6 = constref(A(6));
std::cout << "a1 = "<< a1;
std::cout << "a2 = "<< a2;
std::cout << "a3 = "<< a3;
std::cout << "a4 = "<< a4;
std::cout << "a5 = "<< a5;
std::cout << "a6 = "<< a6;
}
I odpowiednie wyjście z użyciem tej samej linii poleceń g++
:
---const A a1 = A(1)
Cstr 1
---const A a2 = A(2).thiz()
Cstr 2
Copy 2
Del 2 2
---const A a3 = constref(A(3))
Cstr 3
Copy 3
Del 3 3
---const A& a4 = A(4)
Cstr 4
---const A& a5 = A(5).thiz()
Cstr 5
Del 5 5
---const A& a6 = constref(A(6))
Cstr 6
Del 6 6
a1 = { *p=1 , v=1 }
a2 = { *p=12 , v=12 }
a3 = { *p=13 , v=13 }
a4 = { *p=4 , v=4 }
a5 = { *p=0 , v=5 }
a6 = { *p=0 , v=6 }
Del 4 4
Del 13 13
Del 12 12
Del 1 1
Uwaga: za pomocą g ++ wersje 4.4 i 4.6, ten fragment zwraca 0 ... – olibre