• <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>

            yehao's Blog

            函數(shù)的調(diào)用規(guī)則(__cdecl,__stdcall,__fastcall,__pascal)

            關(guān)于函數(shù)的調(diào)用規(guī)則(調(diào)用約定),大多數(shù)時候是不需要了解的,但是如果需要跨語言的編程,比如VC寫的dll要delphi調(diào)用,則需要了解。

                    microsoft的vc默認的是__cdecl方式,而windows API則是__stdcall,如果用vc開發(fā)dll給其他語言用,則應(yīng)該指定__stdcall方式。堆棧由誰清除這個很重要,如果是要寫匯編函數(shù)給C調(diào)用,一定要小心堆棧的清除工作,如果是__cdecl方式的函數(shù),則函數(shù)本身(如果不用匯編寫)則不需要關(guān)心保存參數(shù)的堆棧的清除,但是如果是__stdcall的規(guī)則,一定要在函數(shù)退出(ret)前恢復(fù)堆棧。
            1.__cdecl
                   所謂的C調(diào)用規(guī)則。按從右至左的順序壓參數(shù)入棧,由調(diào)用者把參數(shù)彈出棧。切記:對于傳送參數(shù)的內(nèi)存棧是由調(diào)用者來維護的。返回值在EAX中因此,對于象printf這樣變參數(shù)的函數(shù)必須用這種規(guī)則。編譯器在編譯的時候?qū)@種調(diào)用規(guī)則的函數(shù)生成修飾名的餓時候,僅在輸出函數(shù)名前加上一個下劃線前綴,格式為_functionname。
            2.__stdcall
                    按從右至左的順序壓參數(shù)入棧,由被調(diào)用者把參數(shù)彈出棧。_stdcall是Pascal程序的缺省調(diào)用方式,通常用于Win32 Api中,切記:函數(shù)自己在退出時清空堆棧,返回值在EAX中。  __stdcall調(diào)用約定在輸出函數(shù)名前加上一個下劃線前綴,后面加上一個“@”符號和其參數(shù)的字節(jié)數(shù),格式為_functionname@number。如函數(shù)int func(int a, double b)的修飾名是_func@12
            3.__fastcall
                   __fastcall調(diào)用的主要特點就是快,因為它是通過寄存器來傳送參數(shù)的(實際上,它用ECX和EDX傳送前兩個雙字(DWORD)或更小的參數(shù),剩下的參數(shù)仍舊自右向左壓棧傳送,被調(diào)用的函數(shù)在返回前清理傳送參數(shù)的內(nèi)存棧)。__fastcall調(diào)用約定在輸出函數(shù)名前加上一個“@”符號,后面也是一個“@”符號和其參數(shù)的字節(jié)數(shù),格式為@functionname@number。這個和__stdcall很象,唯一差別就是頭兩個參數(shù)通過寄存器傳送。注意通過寄存器傳送的兩個參數(shù)是從左向右的,即第一個參數(shù)進ECX,第2個進EDX,其他參數(shù)是從右向左的入stack。返回仍然通過EAX.
            4.__pascal
                   這種規(guī)則從左向右傳遞參數(shù),通過EAX返回,堆棧由被調(diào)用者清除

             

            5.__thiscall

                    僅僅應(yīng)用于"C++"成員函數(shù)。this指針存放于CX寄存器,參數(shù)從右到左壓。thiscall不是關(guān)鍵詞,因此不能被程序員指定

             

                   調(diào)用約定可以通過工程設(shè)置:Setting...\C/C++ \Code Generation項進行選擇,缺省狀態(tài)為__cdecl。

             

            名字修飾約定:

            1、修飾名(Decoration name):"C"或者"C++"函數(shù)在內(nèi)部(編譯和鏈接)通過修飾名識別
            2、C編譯時函數(shù)名修飾約定規(guī)則:
            __stdcall調(diào)用約定在輸出函數(shù)名前加上一個下劃線前綴,后面加上一個"@"符號和其參數(shù)的字節(jié)數(shù),格式為_functionname@number,例如 :function(int a, int b),其修飾名為:_function@8
            __cdecl調(diào)用約定僅在輸出函數(shù)名前加上一個下劃線前綴,格式為_functionname。
            __fastcall調(diào)用約定在輸出函數(shù)名前加上一個"@"符號,后面也是一個"@"符號和其參數(shù)的字節(jié)數(shù),格式為@functionname@number。

            3、C++編譯時函數(shù)名修飾約定規(guī)則:
            __stdcall調(diào)用約定:
            1)、以"?"標(biāo)識函數(shù)名的開始,后跟函數(shù)名;
            2)、函數(shù)名后面以"@@YG"標(biāo)識參數(shù)表的開始,后跟參數(shù)表;
            3)、參數(shù)表以代號表示:
            X--void ,
            D--char,
            E--unsigned char,
            F--short,
            H--int,
            I--unsigned int,
            J--long,
            K--unsigned long,
            M--float,
            N--double,
            _N--bool,
            PA--表示指針,后面的代號表明指針類型,如果相同類型的指針連續(xù)出現(xiàn),以"0"代替,一個"0"代表一次重復(fù);
            4)、參數(shù)表的第一項為該函數(shù)的返回值類型,其后依次為參數(shù)的數(shù)據(jù)類型,指針標(biāo)識在其所指數(shù)據(jù)類型前;
            5)、參數(shù)表后以"@Z"標(biāo)識整個名字的結(jié)束,如果該函數(shù)無參數(shù),則以"Z"標(biāo)識結(jié)束。
            其格式為"?functionname@@YG*****@Z"或"?functionname@@YG*XZ",例如
                      int Test1(char *var1,unsigned long)-----“?Test1@@YGHPADK@Z”
                      void Test2()                       -----“?Test2@@YGXXZ

             

            __cdecl調(diào)用約定:
            規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開始標(biāo)識由上面的"@@YG"變?yōu)?@@YA"。
            __fastcall調(diào)用約定:
            規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開始標(biāo)識由上面的"@@YG"變?yōu)?@@YI"。
            VC++對函數(shù)的省缺聲明是"__cedcl",將只能被C/C++調(diào)用.


            注意:
            1、_beginthread需要__cdecl的線程函數(shù)地址,_beginthreadex和CreateThread需要__stdcall的線程函數(shù)地址。

            2、一般WIN32的函數(shù)都是__stdcall。而且在Windef.h中有如下的定義:
            #define CALLBACK __stdcall
            #define WINAPI  __stdcall

            3、extern "C" _declspec(dllexport) int __cdecl Add(int a, int b);
               typedef int (__cdecl*FunPointer)(int a, int b);
               修飾符的書寫順序如上。

            4、extern "C"的作用:如果Add(int a, int b)是在c語言編譯器編譯,而在c++文件使用,則需要在c++文件中聲明:extern "C" Add(int a, int b),因為c編譯器和c++編譯器對函數(shù)名的解釋不一樣(c++編譯器解釋函數(shù)名的時候要考慮函數(shù)參數(shù),這樣是了方便函數(shù)重載,而在c語言中不存在函數(shù)重載的問題),使用extern "C",實質(zhì)就是告訴c++編譯器,該函數(shù)是c庫里面的函數(shù)。如果不使用extern "C"則會出現(xiàn)鏈接錯誤。
            一般象如下使用:
            #ifdef _cplusplus
            #define ETERN_C extern "C"
            #else
            #define EXTERN_C extern
            #endif

            #ifdef _cplusplus
            extern "C"{
            #endif
            EXTERN_C int func(int a, int b);
            #ifdef _cplusplus
            }
            #endif

            5、MFC提供了一些宏,可以使用AFX_EXT_CLASS來代替__declspec(DLLexport),并修飾類名,從而導(dǎo)出類,AFX_API_EXPORT來修飾函數(shù),AFX_DATA_EXPORT來修飾變量
            AFX_CLASS_IMPORT:__declspec(DLLexport)
            AFX_API_IMPORT:__declspec(DLLexport)
            AFX_DATA_IMPORT:__declspec(DLLexport)
            AFX_CLASS_EXPORT:__declspec(DLLexport)
            AFX_API_EXPORT:__declspec(DLLexport)
            AFX_DATA_EXPORT:__declspec(DLLexport)
            AFX_EXT_CLASS:#ifdef _AFXEXT
                AFX_CLASS_EXPORT
                    #else
                AFX_CLASS_IMPORT

            6、DLLMain負責(zé)初始化(Initialization)和結(jié)束(Termination)工作,每當(dāng)一個新的進程或者該進程的新的線程訪問DLL時,或者訪問DLL的每一個進程或者線程不再使用DLL或者結(jié)束時,都會調(diào)用DLLMain。但是,使用TerminateProcess或TerminateThread結(jié)束進程或者線程,不會調(diào)用DLLMain。

            7、一個DLL在內(nèi)存中只有一個實例
            DLL程序和調(diào)用其輸出函數(shù)的程序的關(guān)系:
            1)、DLL與進程、線程之間的關(guān)系
            DLL模塊被映射到調(diào)用它的進程的虛擬地址空間。
            DLL使用的內(nèi)存從調(diào)用進程的虛擬地址空間分配,只能被該進程的線程所訪問。
            DLL的句柄可以被調(diào)用進程使用;調(diào)用進程的句柄可以被DLL使用。
            DLLDLL可以有自己的數(shù)據(jù)段,但沒有自己的堆棧,使用調(diào)用進程的棧,與調(diào)用它的應(yīng)用程序相同的堆棧模式。

            2)、關(guān)于共享數(shù)據(jù)段
            DLL定義的全局變量可以被調(diào)用進程訪問;DLL可以訪問調(diào)用進程的全局數(shù)據(jù)。使用同一DLL的每一個進程都有自己的DLL全局變量實例。如果多個線程并發(fā)訪問同一變量,則需要使用同步機制;對一個DLL的變量,如果希望每個使用DLL的線程都有自己的值,則應(yīng)該使用線程局部存儲(TLS,Thread Local Strorage)。

             

            本文來自CSDN博客,轉(zhuǎn)載請標(biāo)明出處:http://blog.csdn.net/jia_xiaoxin/archive/2008/09/02/2868216.aspx

            posted on 2011-05-15 21:59 厚積薄發(fā) 閱讀(353) 評論(0)  編輯 收藏 引用 所屬分類: C/C++

            導(dǎo)航

            <2025年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            統(tǒng)計

            常用鏈接

            留言簿

            隨筆分類

            文章分類

            文章檔案

            搜索

            最新評論

            久久精品国产色蜜蜜麻豆| 精品久久久久中文字| 久久精品人妻中文系列| 久久中文字幕精品| 国产精品美女久久久久| 91精品国产高清久久久久久91 | 精品999久久久久久中文字幕| 国产精品激情综合久久| 亚洲AV无码成人网站久久精品大| 久久se精品一区精品二区| 欧美色综合久久久久久 | 蜜臀av性久久久久蜜臀aⅴ麻豆| 久久九色综合九色99伊人| 久久久九九有精品国产| 亚洲成色www久久网站夜月| 久久电影网一区| 国产∨亚洲V天堂无码久久久| 亚洲午夜久久久| 一本色道久久综合| 国产一区二区精品久久岳| 97精品国产91久久久久久| 精品乱码久久久久久久| 久久久久久国产a免费观看黄色大片| 色综合久久精品中文字幕首页| 久久福利资源国产精品999| 久久SE精品一区二区| 久久精品国产99久久香蕉| 日韩精品国产自在久久现线拍| 久久精品99久久香蕉国产色戒 | 狠狠久久亚洲欧美专区| 国产激情久久久久久熟女老人| 久久久久高潮毛片免费全部播放 | 91精品无码久久久久久五月天| 精品国际久久久久999波多野| 少妇人妻88久久中文字幕| 亚洲中文久久精品无码ww16| 少妇高潮惨叫久久久久久| 久久99精品国产麻豆| 一本久久久久久久| 精品久久人人爽天天玩人人妻 | 久久综合久久综合九色|