看板 C_and_CPP 關於我們 聯絡資訊
c++新手最看了一些文章,發現一個問題,舉個例子: #include <iostream> class A{ public: int a=5; int& g(){ return a; } A f(){ return *this; } }; int main() { A obj; std::cout<<obj.f().g(); return 0;} 因為obj.f()的lifetime會持續到 std::cout<<obj.f().g(); 這個line結束,所以g取obj.f().a的reference是ok的 但我還是覺得很疑惑,對temporary object 取reference 一般是不行的吧? 舉例來說 int foo1(int a){ return a;} int main(){ int& b=foo(5);} 如果說foo(5)會存在直到 int& b=foo(5) 這行結束,那這個code不是應該也ok嗎 或是 int& foo2(int a){ return a;} int main(){ int b=foo2(5); std::cout<<foo2(5);} 同理這個不是也應該ok了嗎.. 這個是可以compile,但是不會cout出東西 把他丟去compiler explorer的話會發現foo2會直接回傳0(?) 我覺得越來越不懂了,求解@@ 謝謝大家 ---- Sent from BePTT on my OPPO CPH1943 -- ※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 1.200.14.226 (臺灣) ※ 文章網址: https://www.ptt.cc/bbs/C_and_CPP/M.1712838430.A.E4C.html
jack7775kimo: 關鍵字:value category04/11 21:24
jack7775kimo: cppreference上應該都能找到解釋04/11 21:28
jack7775kimo: 以上是回答第二個問題;最初那個obj.f()問題是因為04/11 21:38
jack7775kimo: RVO,所以obj.f()會拿到obj.a 不是暫時物件04/11 21:39
jack7775kimo: 更正:就算compiler因沒有RVO而拿到暫時物件,也無妨04/11 21:44
謝謝! 那如果f改成{A obj2; return obj2;}還會有RVO嗎(改成這樣output跟原來一樣)
Dracarys: 1. int&可以bind到A::g中的a,因為a是lvalue04/11 21:46
謝謝! 1.的 a是lvalue? 但是obj.f() 應該是temporary 那就不應該是lvalue不是嗎0.0
Dracarys: 2. int& b = foo1(5)違法是因為foo1(5)是prvalue (pure04/11 21:46
嗯嗯所以lvalue reference應該還是是不能bind to xvalue 或是prvalue囉xd(混亂
Dracarys: rvalue)04/11 21:46
Dracarys: 3. foo2在C++20及以前都編得過,但是return的reference04/11 21:46
Dracarys: 是dangle的,去印出來是未定義行為。C++23 P2266R3開始04/11 21:46
Dracarys: ,a作為return的operand是xvalue,不能被bound to non-04/11 21:46
Dracarys: const lvalue reference,int&改成int&&或const int&才04/11 21:46
Dracarys: 編得過。04/11 21:46
Dracarys: http://wg21.link/P2266R304/11 21:46
wulouise: 通常guideline是永遠不要這麼做,比較節省維護心力04/11 22:10
謝謝,也是沒有書會這樣寫xd ※ 編輯: amamoimi (1.200.14.226 臺灣), 04/11/2024 22:35:00
Dracarys: Non-const lvalue reference只能bind to lvalue04/11 23:10
Dracarys: 不能rvalue (含xvalue、prvalue)04/11 23:10
Dracarys: a是不是lvalue跟obj.f()沒關係,通常寫一個變數名都是l04/11 23:16
Dracarys: value expression ,例外我只想得到C++23改的那個case04/11 23:16
也就是說obj.f()本來是rvalue,但是到g裡就被當成成lvalue了嗎!?為什麼會這樣...
jack7775kimo: 要看compiler廠商怎麼實作,但一般來說你這樣寫會有04/12 07:53
amamoimi: 不知道為什麼我編輯之後推文整個亂掉了...j大的推文好04/12 09:57
amamoimi: 像被截掉了,真的很不好意思@@04/12 09:57
※ 編輯: amamoimi (180.217.14.80 臺灣), 04/12/2024 10:19:43
jack7775kimo: 1)更正前述的obj.a應是obj 04/12 11:38
jack7775kimo: 2) 改code,意圖也跟著變了;原先f是拿到obj自己,新的 04/12 11:38
jack7775kimo: 是拿到obj的複製體. (但這不影響有沒有RVO) 04/12 11:40
amamoimi: 謝謝j大復原! ※ 編輯: amamoimi (140.138.31.178 臺灣), 04/12/2024 12:56:17
Dracarys: Value category是expression的屬性,obj.f()跟a是不同e 04/12 16:30
Dracarys: xpr.所以我前面才說沒關係 04/12 16:30
amamoimi: 那如果我說: obj.f()是temporary,那以它呼叫g()時,g 04/12 16:59
amamoimi: 會把這個temporary object 當成有名字的物件,是嗎xd 04/12 16:59
LPH66: 再說一次, value category 是式子的屬性, 不是物件的屬性 04/13 14:36
LPH66: 即使在不同狀況下參照到同一個物件仍然可能是不同 category 04/13 14:37
LPH66: (因此才會有延長物件壽命的規則, 因為 category 可能變化) 04/13 14:38
firejox: 可以先看這篇 #19gioP8j (C_and_CPP) 這樣會比較好理解 04/13 23:06
amamoimi: 謝謝文章!我覺得我應該懂我第一個例子為什麼valid了.. 04/14 06:49
amamoimi: .(其實學校學的c++都還是很原始的版本,但是每次遇到 04/14 06:49
amamoimi: 問題,好像都需要用到c++11(?的觀念去解決..)另外雖 04/14 06:49
amamoimi: 然value指的是expression,但是我怎麼覺得有好幾個地方 04/14 06:49
amamoimi: 他指的都是物件啊(或是參數、運算元,總之就是一個名 04/14 06:49
amamoimi: 詞(?) 04/14 06:49
firejox: 因為要考慮到123、true等常數的情況,所以表達上用value 04/14 19:20