[轉]你可能不知道的C++(一)

此為《你可能不知道的C++》的第一部分,討論C&c++,編譯單元以及對象

C++&c##

C++ 是 C 的超集,但是 C++ 中的子集 C 跟原始的 C 還是有點不一樣。

結構 & 聯合###

  • C 的結構(struct)不是一種類型,使用時得帶著關鍵字struct,一般用typedef來避免這種不便。
  • C++ 的結構幾乎等價于類,只是缺省的訪問權限為public而非private。
  • C++ 的聯合(union)可以有成員函數,甚至可以有構造和析構函數。

不帶參數的函數###

對 C 來說,一個不帶參數的函數意味著可以接受任意參數。所以void f()就相當于void f(...),而下面三個函數指針類型中:
<pre>typedef void (foox)();
typedef void (
foo1)(int);
typedef void (*foo2)(void);
</pre>

foo1和foo2可以隱式地轉型為foox,就好比可以從int或char隱式地轉型成void*。

要想讓一個 C 函數真正沒有參數,得用void:
<pre>void foo(void);</pre>

對 C++ 來說,一個不帶參數的函數就是指不接受參數。往參數列表里放void是多余的。

提升void*###

C 會自動提升(promote)void
<pre>int
pi = malloc(sizeof(int));</pre>

函數malloc返回void,賦值給int時不需要顯式轉型。而 C++ 必須顯式轉型:
<pre>int* pi = static_cast
(int*)(malloc(sizeof(int)));</pre>

CONSTS###

C++ 允許 consts 用在常量表達式中:

<pre>const int MAX = 4;
int a[MAX + 1];
switch (i) {
case MAX:
...
}</pre>

而 C 則必須使用宏:
<pre>#define MAX 4</pre>

引一段《C++ 的設計和演化》的原文:
(Bjarne Stroustrup, The Design and Evolution Of C++, 3.8)

In C, consts may not be used in constant expressions. This makes consts far less useful in C than in C++ and leaves C dependent on the preprocessror while C++ programmers can use properly typed and scoped consts.
</br>

C 的 consts(特指用 const 關鍵字修飾的常量)不可以用在常量表達式中。這讓 C 的 consts 遠不如 C++ 的有用,也讓 C 依賴于預處理器,而 C++ 程序員則可以使用有適當類型和作用域的 consts。

前置聲明###

C 代碼塊中,所有聲明必須出現在任何程序語句之前,比如函數定義時,先聲明所有局部變量:
<pre>
void foo() {
int ival, p;
/
… */
}
</pre>

而 C++ 的聲明,諸如int ival;,其自身就是一個程序語句,也因此可以出現在程序文本中的任何位置。

編譯單元##

C/C++ 中的一個源文件(.c, .cpp, .cc)就是一個編譯單元(compilation unit)。
頭文件(.h, .hpp)不是編譯單元,是不能單獨編譯的。

源文件經過預處理,先搞定下面這些東西:

  • 宏:包括用戶定義的,和預定義的(__cplusplus, FILE, ...)
  • 包含語句:源文件中的include語句全部展開
  • 條件編譯: #if, #else, #ifudef, ...#error, #warning, ...

預處理過的源文件,經過編譯,生成對象文件(.o, .obj)。對象文件經過鏈接或打包,生成可執行文件或程序庫。雖然這里的步驟不太嚴格,但是大抵就是這樣。

如果你對預處理的結果很感興趣,可以試試編譯器的預處理命令:gcc -E (GCC),cl /E or /P (VC)。

對象##

這里所說的對象(object),泛指一切類型的實例,不只是類的實例。

關于對象,我們將探討以下幾個方面:

  • 對象的大小(size)
  • 按存儲(storage)分類的對象
  • 聚合(aggregate)

對象的大小###

先來考慮幾個問題:

  • sizeof是一個函數嗎?
  • 你知道sizeof(int), sizeof(long)各為多少嗎?
  • 為什么應該用size_t?

<strong>size_t</strong>

標準庫里到處都是size_t的身影:
<pre>
void *malloc(size_t n);
void *memcpy(void *s1, void const *s2, size_t n);
size_t strlen(char const *s);
</pre>

回到前面的問題,不難理解以下幾點:

  • size_t是sizeof返回值的類型
  • size_t是一個typedef
  • sizeof不是一個函數,它是一個編譯時操作符
  • size_t能夠表示任何類型理論上可能的數組的最大大小

其實,size_t一般就是unsigned int的typedef,那為什么不直接用unsigned int?在IP16或IP32平臺上(即int和指針大小一致時),確實沒有問題,但I16LP32就不行了。此外,直接用unsigned long固然沒錯,但畢竟得多花了幾個字節,稍微有點浪費了。反正只要用size_t,你就可以同時得到正確性和可移植性。

<strong>數據對齊</strong>

請問mixed_data的大小是多少?是 8 嗎?
<pre>
struct mixed_data {
char data1;
short data2;
int data3;
char data4;
};
</pre>

在 32 位 x86 平臺上編譯后的樣子:
<pre>
struct mixed_data {
char data1;
char padding1[1];
short data2;
int data3;
char data4;
char padding2[3];
};</pre>
為了數據對齊,編譯器塞了一些邊角料進去,最終的大小為 12

<strong>按存儲分類的對象</strong>
C/C++ 的對象,按存儲類型分為以下幾種:

  • 自動的(auto, register)</br>
  • 靜態的(static)
    自由存儲的(free-store)
    關鍵字auto有點多余,下面兩條聲明語句其實等價,b前面的auto加不加一個效果:
    <pre>
    {
    int a;
    auto int b;
    }
    </pre>
    到了 C++11,auto這個關鍵字就被拿來另作他用了:auto可以讓編譯器從變量的初始化上自動推斷出它的類型:
    <pre>auto a = std::max(1.0, 4.0); // 編譯器推斷出 a 的類型為 double</pre>

聚合###

首先,什么叫聚合?

對 C 來說,數組和結構是聚合。

對 C++ 來說,除了數組外,滿足以下條件的類(或結構)也是聚合:

  • 沒有用戶聲明的構造函數
  • 沒有private或protected非靜態數據成員
  • 沒有基類
  • 沒有虛函數

所以,下面幾個類型都是聚合:
<pre>
int[5];

struct person {
std::string name;
int age;
};

boost::array;
</pre>

<strong>第一部分完。</strong>
原文地址

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,619評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,155評論 3 425
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,635評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,539評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,255評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,646評論 1 326
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,655評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,838評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,399評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,146評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,338評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,893評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,565評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,983評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,257評論 1 292
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,059評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,296評論 2 376

推薦閱讀更多精彩內容