• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            main函數(shù)之前干了神馬?-誰(shuí)調(diào)用了我的main

            (轉(zhuǎn)載自http://topic.csdn.net/u/20101013/16/e5f2fcbb-528f-4b1a-b85c-aaa8aa9ab3b6.html
            main函數(shù)之前--真正的函數(shù)執(zhí)行入口或開(kāi)始

            一種解釋

            實(shí)際上,在可執(zhí)行文件被加載之后,控制權(quán)立即交給由編譯器插入的Start函數(shù),它將對(duì)后面這些全局變量進(jìn)行準(zhǔn)備:
               _osver 操作系統(tǒng)的構(gòu)件編號(hào)
              _winmajor 操作系統(tǒng)的主版本號(hào)
              _winminor 操作系統(tǒng)的次版本號(hào)
              _winver 操作系統(tǒng)完全版本號(hào)
              __argc 命令行參數(shù)個(gè)數(shù)
              __argv 指向參數(shù)字符串的指針數(shù)組
              _environ 指向環(huán)境變量字符串的指針數(shù)組
            Start函數(shù)初始化堆并調(diào)用main函數(shù).mian函數(shù)返回之后,Start函數(shù)調(diào)用Exit函數(shù)結(jié)束該進(jìn)程.
            啟動(dòng)函數(shù)Start的源代碼在:
               crt0.c Microsoft Visual C++
              c0w.asm Borladn C++

            另一種解釋

            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

            第三種解釋:囫圇C語(yǔ)言(三):誰(shuí)調(diào)用了我的 main?
                
                現(xiàn)在最重要的是要跟得上潮流,所以套用比較時(shí)髦的話,誰(shuí)動(dòng)了我的奶酪。誰(shuí)調(diào)用了我的 main?不過(guò)作為計(jì)算機(jī)工作者,我勸大家還是不要趕時(shí)髦,今天Java熱,明天 .net 流行,什么時(shí)髦就學(xué)什么。我的意思是先花幾年把基本功學(xué)好,等你趕時(shí)髦的時(shí)候也好事半功倍。廢話不多說(shuō)了。
                
                我們都聽(tīng)說(shuō)過(guò)一句話:“main是C語(yǔ)言的入口”。我至今不明白為什么這么說(shuō)。就好像如果有人說(shuō):“掙錢是泡妞”,肯定無(wú)數(shù)磚頭拍過(guò)來(lái)。這句話應(yīng)該是“掙錢是泡妞的一個(gè)條件,只不過(guò)這個(gè)條件特別重要”。那么上面那句話應(yīng)該是 “main是C語(yǔ)言中一個(gè)符號(hào),只不過(guò)這個(gè)符號(hào)比較特別。”
                
                我們看下面的例子:
                
                /* file name test00.c */
                
                int main(int argc, char* argv)
                {
                 return 0;
                }
                
                編譯鏈接它:
                cc test00.c -o test.exe
                會(huì)生成 test.exe
                
                但是我們加上這個(gè)選項(xiàng): -nostdlib (不鏈接標(biāo)準(zhǔn)庫(kù))
                cc test00.c -nostdlib -o test.exe
                鏈接器會(huì)報(bào)錯(cuò):
                undefined symbol: __start
                
                也就是說(shuō):
                1. 編譯器缺省是找 __start 符號(hào),而不是 main
                2. __start 這個(gè)符號(hào)是程序的起始點(diǎn)
                3. main 是被標(biāo)準(zhǔn)庫(kù)調(diào)用的一個(gè)符號(hào)
                
                再來(lái)思考一個(gè)問(wèn)題:
                我們寫(xiě)程序,比如一個(gè)模塊,通常要有 initialize 和 de-initialize,但是我們寫(xiě) C 程序的時(shí)候?yàn)槭裁从行┠K沒(méi)有這兩個(gè)過(guò)程么呢?比如我們程序從 main 開(kāi)始就可以 malloc,free,但是我們?cè)?main 里面卻沒(méi)有初始化堆。再比如在 main 里面可以直接 printf,可是我們并沒(méi)有打開(kāi)標(biāo)準(zhǔn)輸出文件啊。(不知道什么是 stdin,stdout,stderr 以及 printf 和 stdout 關(guān)系的群眾請(qǐng)先看看 C 語(yǔ)言中文件的概念)。
                
                有人說(shuō),這些東西不需要初始化。如果您真得這么想,請(qǐng)您不要再往下看了,我個(gè)人認(rèn)為計(jì)算機(jī)軟件不適合您。
                
                聰明的人民群眾會(huì)想,一定是在 main 之前干了些什么。使這些函數(shù)可以直接調(diào)用而不用初始化。通常,我們會(huì)在編譯器的環(huán)境中找到一個(gè)名字類似于 crt0.o 的文件,這個(gè)文件中包含了我們剛才所說(shuō)的 __start 符號(hào)。(crt 大概是 C Runtime 的縮寫(xiě),請(qǐng)大家?guī)椭_認(rèn)一下。)
                
                那么真正的 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;
                ////////////////////////////////////////////////////
                
                實(shí)際上可能還有很多初始化工作,因?yàn)槎际呛筒僮飨到y(tǒng)相關(guān)的,筆者就不一一列出了。
                
                注意:
                1. 不同的編譯器,不一定缺省得符號(hào)都是 __start。
                2. 匯編里面的 _main 就是 C 語(yǔ)言里面的 main,是因?yàn)閰R編器和C編譯器對(duì)符號(hào)的命名有差異(通常是差一個(gè)下劃線'_')。
                3. 目前操作系統(tǒng)結(jié)構(gòu)有兩個(gè)主要的分支:微內(nèi)核和宏內(nèi)核。微內(nèi)核的優(yōu)點(diǎn)是,結(jié)構(gòu)清晰,簡(jiǎn)單,內(nèi)核組件較少,便于維護(hù);缺點(diǎn)是,進(jìn)程間通信較多,程序頻繁進(jìn)出內(nèi)核,效率較低。宏內(nèi)核正好相反。我說(shuō)這個(gè)是什么目的是:沒(méi)辦法保證每個(gè)組件都在用戶空間(標(biāo)準(zhǔn)庫(kù)函數(shù))中初始化,有些組件確實(shí)可能不要初始化,操作系統(tǒng)在創(chuàng)建進(jìn)程的時(shí)候在內(nèi)核空間做的。這依賴于操作系統(tǒng)的具體實(shí)現(xiàn),比如堆,宏內(nèi)核結(jié)構(gòu)可能在內(nèi)核初始化,微內(nèi)核結(jié)構(gòu)在用戶空間;即使同樣是微內(nèi)核,這個(gè)東東也可能會(huì)被拿到內(nèi)核空間初始化。
                
                隨著 CPU 技術(shù)的發(fā)展,存儲(chǔ)量的迅速擴(kuò)展,代碼復(fù)雜程度的增加,微內(nèi)核被越來(lái)越多的采用。你會(huì)為了 10% 的效率使代碼復(fù)雜度增加么?要知道每隔 18 個(gè)月 CPU 的速度就會(huì)翻一番。所以我對(duì)程序員的要求是,我首先不要你的代碼效率高,我首先要你的代碼能讓 80% 的人迅速看懂并可以維護(hù)。

            總結(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.運(yùn)行全局構(gòu)造器,估計(jì)是C++中構(gòu)造函數(shù)之類的吧

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

            posted on 2011-03-08 23:34 MrRightLeft 閱讀(2698) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C/C++

            <2011年3月>
            272812345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            導(dǎo)航

            統(tǒng)計(jì)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            搜索

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久久噜噜噜久久中文字幕色伊伊 | 久久最新免费视频| 青青热久久国产久精品| 中文无码久久精品| 久久精品国产精品国产精品污| 久久国产三级无码一区二区| 亚洲欧美国产日韩综合久久| 97久久香蕉国产线看观看| 久久无码一区二区三区少妇| 无码人妻精品一区二区三区久久久| 久久久91精品国产一区二区三区 | 久久综合精品国产二区无码| 久久亚洲国产中v天仙www| 2019久久久高清456| 久久青草国产手机看片福利盒子| 久久久久亚洲av成人网人人软件| 久久精品国产亚洲欧美| 中文字幕无码免费久久| 香蕉久久永久视频| 国产激情久久久久影院| 国产亚洲精品自在久久| 久久久久亚洲精品日久生情| 色播久久人人爽人人爽人人片aV| 日韩亚洲欧美久久久www综合网| 午夜精品久久久久久毛片| 中文字幕精品无码久久久久久3D日动漫 | 精品久久久噜噜噜久久久| 久久强奷乱码老熟女网站| 久久精品无码一区二区三区免费| 久久国产精品一区二区| 国产精品久久毛片完整版| 久久精品国产99久久无毒不卡| 国产色综合久久无码有码| 久久久亚洲欧洲日产国码是AV | 久久夜色撩人精品国产| 狠狠色丁香婷婷综合久久来来去 | 香蕉久久影院| 一本一本久久A久久综合精品 | 久久亚洲精品视频| 亚洲午夜精品久久久久久人妖| 日韩精品久久久久久|