c语言编译的动态链接库中,导出的符号名字就是 源代码中的相应的名字;
例如函数 void test(){} 导出的符号名字 就是 test
但是对于c++ 来讲不是这样, c++ 有类, 名字空间, 函数重载, 导致多个不同的对象可能使用一个相同的名字, 这样必须由编译器来生成全局的唯一名字;
这种生成的方式 没有标准化, 所以 window上的vc 编译器可能生成一种名字, linux上的gcc 可能生成一种名字, mingw 可能也会生成一种名字;
因此不同编译器生成的C++动态库, 从符号名字上来讲,不兼容。
但是如果需要从C++库导出某些名字 使得大家都认识该怎么做呢?
源代码中写下:
extern "c"{
void test();
}
将会生成 标准的c符号名字, 也就是 test, 这样所有人都会认识这个符号了!
当然要注意一点 所有声明该符号的位置 不能出现在名字空间中, 否则 仍旧生成 C++类型的符号名字。
标准c中定义了函数的参数的压栈和出栈相关问题: 压栈从右向左压入参数;而当函数返回的时候, 由调用者负责清理堆栈, 弹出 相应的参数
也就是被调用者 直接return, 调用者再清理 堆栈顶部指针, 这样实现printf 函数就比较方便了(可变长度参数, 被调用者不用关心参数的数量)
但是win32中, 有一种windows专用的stdcall方法, 参数也是从右向左压入, 但是 函数返回的时候是由被调用者清理堆栈
也就是被调用者先拿到返回地址, 在设置堆栈顶部指针, 最后返回(被调用者需要关心参数的数量)
这些是windows 需要的东西:
因此在写win32程序的时候 如果需要 采用标准的 c参数处理方式, 就需要显示的告诉 编译器;
__cdecl 标准c的参数管理方式 __stdcall windows的参数管理方式
主要用于dll说明参数的管理方式的:
__declspec(dllexport)用于导出 该动态库 的标准c方式的函数
__declspec(dllimport) 用于一个动态库 依赖于 另一个动态库的函数