C語言深度總結(jié)[全面認(rèn)識main函數(shù)之前運(yùn)行代碼]

一锯岖、main運(yùn)行前可運(yùn)行哪些代碼

? (1)全局對象的構(gòu)造函數(shù)會在main 函數(shù)之前執(zhí)行。

? (2)一些全局變量痹兜、對象和靜態(tài)變量、對象的空間分配和賦初值就是在執(zhí)行main函數(shù)之前颤诀,而main函數(shù)執(zhí)行完后字旭,還要去執(zhí)行一些諸如釋放空間、釋放資源使用權(quán)等操作

? (3)進(jìn)程啟動后着绊,要執(zhí)行一些初始化代碼(如設(shè)置環(huán)境變量等)谐算,然后跳轉(zhuǎn)到main執(zhí)行。全局對象的構(gòu)造也在main之前归露。

? (4)通過關(guān)鍵字__attribute__洲脂,讓一個函數(shù)在主函數(shù)之前運(yùn)行,進(jìn)行一些數(shù)據(jù)初始化、模塊加載驗證等恐锦。

main函數(shù)運(yùn)行之前需要運(yùn)行的邏輯(為了保持原味往果,我就不翻譯了)

Some of the stuff that has to happen before main():

set up initial stack pointer?

initialize static and global data?

zero out uninitialized data?

run global constructors

Some of this comes with the runtime library's crt0.o file or its __start() function. Some of it you need to do yourself.

Crt0 is a synonym for the C runtime library.

Depending on the system you're using the follwing may be incomplete, but it should give you an idea. Using newlib-1.9.0/libgloss/m68k/crt0.S as an outline, the steps are:

1. Set stack pointer to value of __STACK if set?

2. Set the initial value of the frame pointer?

3. Clear .bss (where all the values that start at zero go)?

4. Call indirect of hardware_init_hook if set to initialize hardware?

5. Call indirect of software_init_hook if set to initialize software?

6. Add __do_global_dtors and __FINI_SECTION__ to the atexit function so destructors and other cleanup functions are called when the program exits by either returning from main, or calling exit

7. setup the paramters for argc, argv, argp and call main?

8. call exit if main returns

誰調(diào)用了我的main函數(shù)

我們都聽說過一句話:“main是C語言的入口”。我至今不明白為什么這么說一铅。就好像如果有人說:“掙錢是泡妞”陕贮,肯定無數(shù)磚頭拍過來。這句話應(yīng)該是“掙錢是泡妞的一個條件潘飘,只不過這個條件特別重要”肮之。那么上面那句話應(yīng)該是 “main是C語言中一個符號,只不過這個符號比較特別卜录「昵埽”

我們看下面的例子:

    /* file name battle.c */

    #include<stdio.h>

    int main(int argc, char* argv[])

    {

????????????????printf("老鐵,感謝有緣一起學(xué)習(xí)C語言艰毒,除了跟著這套課程走筐高,為了更好地解決大家實操中的問題,可以加QQ群676593534丑瞧,及時交流柑土。群里還有很多精致資料哦,我這這里等你绊汹!");

     return 0;

    }

編譯鏈接它:

cc battle.c -o test.exe

會生成 test.exe

但是我們加上這個選項: -nostdlib (不鏈接標(biāo)準(zhǔn)庫)

cc battle.c-nostdlib -o test.exe

鏈接器會報錯:

undefined symbol: __start

也就是說:

1. 編譯器缺省是找 __start 符號稽屏,而不是 main

2. __start 這個符號是程序的起始點

3. main 是被標(biāo)準(zhǔn)庫調(diào)用的一個符號

繼續(xù)探索

 我們寫程序,比如一個模塊灸促,通常要有 initialize 和 de-initialize诫欠,但是我們寫 C 程序的時候為什么有些模塊沒有這兩個過程么呢?比如我們程序從 main 開始就可以 malloc浴栽,free荒叼,但是我們在 main 里面卻沒有初始化堆。再比如在 main 里面可以直接 printf典鸡,可是我們并沒有打開標(biāo)準(zhǔn)輸出文件啊被廓。(不知道什么是 stdin,stdout萝玷,stderr 以及 printf 和 stdout 關(guān)系的群眾請先看看 C 語言中文件的概念)嫁乘。

有人說,這些東西不需要初始化球碉。如果你真得這么想蜓斧,有點率性而為了。

  聰明的人民群眾會想睁冬,一定是在 main 之前干了些什么挎春。使這些函數(shù)可以直接調(diào)用而不用初始化。通常,我們會在編譯器的環(huán)境中找到一個名字類似于 crt0.o 的文件直奋,這個文件中包含了我們剛才所說的 __start 符號能庆。(crt 大概是 C Runtime 的縮寫)

那么真正的 crt0.s 是什么樣子呢?下面我們給出部分偽代碼:

    section .text:

    __start:

     :

     init stack;

     init heap;

     open stdin;

     open stdout;

     open stderr;

     :

     push argv;

     push argc;

     call _main; (調(diào)用 main)

     :

     destory heap;

     close stdin;

     close stdout;

     close stderr;

     :

     call __exit;

    實際上可能還有很多初始化工作脚线,因為都是和操作系統(tǒng)相關(guān)的搁胆,這里就不一一列出了。

注意:

1. 不同的編譯器邮绿,不一定缺省得符號都是 __start渠旁。

2. 匯編里面的 _main 就是 C 語言里面的 main,是因為匯編器和C編譯器對符號的命名有差異(通常是差一個下劃線'_')船逮。

Demo演示

#include <stdio.h>

#if 0

The constructor attribute causes the function to be called automatically before?

execution enters main (). Similarly, the destructor attribute causes the function?

to be called automatically after main () completes or exit () is called.?

Functions with these attributes are useful for initializing data that is used?

implicitly during the execution of the program.

more infoformation: https://gcc.gnu.org/onlinedocs/gcc-6.2.0/gcc/Common-Function-Attributes.html#Common-Function-Attributes

#endif?


__attribute__((constructor)) void before_main_to_run()?

{?

? ? printf("Hi~,i am called before the main function!\n");

? ? printf("%s\n",__FUNCTION__);?

}?

__attribute((constructor)) void before_main_to_run_two()?

{?

? ? printf("Hi~,i am called before the main function!\n");

? ? printf("%s\n",__FUNCTION__);?

}?


__attribute__((destructor)) void after_main_to_run()?

{?

? ? printf("%s\n",__FUNCTION__);?

? ? printf("Hi~,i am called after the main function!\n");

}?


__attribute((destructor)) void after_main_to_run_two()?

{?

? ? printf("%s\n",__FUNCTION__);?

? ? printf("Hi~,i am called after the main function!\n");

}?

int main( int argc, char ** argv )?

{?

? ? printf("i am main function, and i can get my name(%s) by this way.\n",__FUNCTION__);?

? ? return 0;?

}?

總結(jié)

main函數(shù)執(zhí)行之前一死,主要就是初始化系統(tǒng)相關(guān)資源:

1.設(shè)置棧指針

2.初始化static靜態(tài)和global全局變量,即data段的內(nèi)容

3.將未初始化部分的賦初值:數(shù)值型short傻唾,int,long等為0承耿,bool為FALSE冠骄,指針為NULL,等等加袋,即.bss段的內(nèi)容

4.將main函數(shù)的參數(shù)凛辣,argc,argv等傳遞給main函數(shù)职烧,然后才真正運(yùn)行main函數(shù)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末扁誓,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子蚀之,更是在濱河造成了極大的恐慌蝗敢,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件足删,死亡現(xiàn)場離奇詭異寿谴,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)失受,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進(jìn)店門讶泰,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人拂到,你說我怎么就攤上這事痪署。” “怎么了兄旬?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵狼犯,是天一觀的道長。 經(jīng)常有香客問我,道長辜王,這世上最難降的妖魔是什么劈狐? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮呐馆,結(jié)果婚禮上肥缔,老公的妹妹穿的比我還像新娘。我一直安慰自己汹来,他們只是感情好续膳,可當(dāng)我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著收班,像睡著了一般坟岔。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上摔桦,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天社付,我揣著相機(jī)與錄音,去河邊找鬼邻耕。 笑死鸥咖,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的兄世。 我是一名探鬼主播啼辣,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼御滩!你這毒婦竟也來了鸥拧?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤削解,失蹤者是張志新(化名)和其女友劉穎富弦,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體氛驮,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡舆声,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了柳爽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片媳握。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖磷脯,靈堂內(nèi)的尸體忽然破棺而出蛾找,到底是詐尸還是另有隱情,我是刑警寧澤赵誓,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布打毛,位于F島的核電站柿赊,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏幻枉。R本人自食惡果不足惜碰声,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽郑象。三九已至贡这,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間厂榛,已是汗流浹背盖矫。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留击奶,地道東北人炼彪。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像正歼,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子拷橘,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容

  • rljs by sennchi Timeline of History Part One The Cognitiv...
    sennchi閱讀 7,325評論 0 10
  • Lua 5.1 參考手冊 by Roberto Ierusalimschy, Luiz Henrique de F...
    蘇黎九歌閱讀 13,793評論 0 38
  • mean to add the formatted="false" attribute?.[ 46% 47325/...
    ProZoom閱讀 2,696評論 0 3
  • 蟲忘記了塵埃局义。 神忘記了永恒。 我還是想堅持自己的生存法則冗疮。 在冷漠的世界里萄唇,深情地活著。 在頹廢的環(huán)境中术幔,熱血地...
    蝸牛36閱讀 111評論 0 0
  • # 老羅語錄 每個生命來到世間另萤,都注定改變世界,別無選擇诅挑。要么變的好一點四敞,要么變的壞一點。你如果走進(jìn)社會為了生存拔妥,...
    行與心田閱讀 624評論 0 0