天天看點

cout 堆棧,operator<< 運算符重載輸出問題

在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;

對于以上過程還需注意兩點:

繼續閱讀