在C++中cout的輸出流其中,有一些問題非常easy出錯,就比方以下這道簡單程式。看似簡單。但卻是一個值得深思的問題~~
#include <iostream>
using namespace std;
int foo(int &x)
{
cout << "第" << x << "次調用" << " ";
return ++x;
}
int main()
int i = 1;
cout << foo(i) << '\t' << foo(i) << endl;
非常多人一開始會以為輸出的結果會是:
第1次調用 2 第2次調用 3
可是結果卻是:
第1次調用 第2次調用 3 2
為什麼嘞??? 首先須要了解的是:
程序空間:
代碼區----------------存放程式的運作代碼
全局資料區------------存放全局資料。常量,文字量。靜态全局量和靜态局部量
堆區------------------存放動态記憶體。供程式随機申請使用
棧區------------------函數資料區(局部資料區)
(以上是執行中的記憶體布局)
函數調用的整個過程就是棧空間操作的過程。
函數調用時,C++作下面工作:
(1)建立被調函數的棧空間。其大小由函數定義體中的資料量決定;
(2)保護調用函數的執行狀态和傳回位址;
(3)傳遞參數;
(4)将控制權轉交給被調函數;
(5)函數執行完畢後。複制傳回植到函數局部資料塊底部;
(6)恢複被調函數的執行狀态;
(7)傳回調用函數。
調用一個函數能夠看作是一個棧中元素的壓棧與退棧操作。
有了以上的介紹,解釋過程就能知道了:
原始輸出語句: cout << foo(i) << '\t' << foo(i) << endl;
cout輸出的運作順序是從右向左運作的,so:
1:運作<<endl,由于沒有std::cout對象,無法運作,壓棧
2:運作<<foo(i),運作foo(1),輸出:第1次調用,得到傳回值2,還是由于沒有std::cout對象。無法立即輸出。壓棧
此時,代碼變成cout << foo(i) << '\t' << 2 << endl;
3:運作<< '\t' 。沒有std::cout對象,無法立即輸出。壓棧
此時。代碼變成cout << foo(i) << '\t' << 2 << endl;
4:運作<<foo(i),運作foo(2)。輸出:第2次調用 得到傳回值3,還是由于沒有std::cout對象,無法立即輸出,壓棧
此時。代碼變成cout << 3 << '\t' << 2 << endl;
對于以上過程還需注意兩點: