您的位置:首页 > 其它

__stdcall和_cdecl FAR&NEAR 16&32

2010-07-10 15:28 204 查看
关于__stdcall和_cdecl     
    
这两个关键字看起来似乎很少和我们打交道,但是看了下面的定义(来自windef.h),你一定会觉得惊讶:

#define CALLBACK __stdcall
#define WINAPI __stdcall
#define WINAPIV
__cdecl
#define APIENTRY WINAPI
#define APIPRIVATE __stdcall
#define
PASCAL __stdcall

#define cdecl _cdecl
#ifndef CDECL
#define
CDECL _cdecl
#endif
几乎我们写的每一个WINDOWS API函数都是__stdcall类型的,为什么??

首先,我们谈一下两者之间的区别:

WINDOWS的函数调用时需要用到栈(STACK,一种先入后出的存储结构)。当函数调用完成后,栈需要清楚,这里就是问题的关键,如何清除??

如果我们的函数使用了_cdecl,那么栈的清除工作是由调用者,用COM的术语来讲就是客户来完成的。这样带来了一个棘手的问题,不同的编译器产生栈的方式不尽相同,那么调用者能否正常的完成清除工作呢?答案是不能。

如果使用__stdcall,上面的问题就解决了,函数自己解决清除工作。所以,在跨(开发)平台的调用中,我们都使用__stdcall(虽然有时是以WINAPI的样子出现)。

那么为什么还需要_cdecl呢?当我们遇到这样的函数如fprintf()它的参数是可变的,不定长的,被调用者事先无法知道参数的长度,事后的清除工作也无法正常的进行,因此,这种情况我们只能使用_cdecl。

到这里我们有一个结论,如果你的程序中没有涉及可变参数,最好使用__stdcall关键字  

near是一般意义的指针,只有16位地址值,只能仅仅寻址64K,可以对指针进行比较和算术运算。

far是一个32位指针,一个far指针包含了段值和偏移量,能访问大于64K的代码段和数据段,由于一个内存位置可以有几个不同的段:偏移值,far指针不可比较和运算。

还有一种huge指针,也是32位,是经过规格化处理的指针,可以进行比较和运算。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息