各位大侠,小弟最近碰到个奇怪问题:在VS2005或者VS2008环境下。创建WIN32 DLL项目。 在测试程序中用动态加载方式即 LoadLibary. 加载成功,但如果DLL项目里函数不声明为extern "C"的话,比如如果声明成_stdcall, 则在测试项目中GetProcAddress()硬是找不到函数名! 关于extern "C" ,_stdcall,_cdecl 的基本概念我有。 我如果把DLL函数声明成_stdcall时, 加载时在typedef里也有加_stdcall,但还是找不到函数(GetLasterror返回127). 所以觉得好奇怪啊!extern "C"就找得到, _stdcall就找不到?

解决方案 »

  1.   

    你用depends查一下你的dll是否有个这个导出函数,有的话应该能找到的,没有自然就找不到了GetProcAddres肯定失败.
      

  2.   

    不声明为extern "C"的话,编译器会对导出函数名字进行改编,如在前面加前导符!所以你找不到。用dumpbin或depends看一下你的dll
      

  3.   

    恩,我是名字改编的问题. 但我的问题是 为什么倒出和调用时加_stdcall保持一致也不行? 难道说如果调用时要写回原来的函数名,非extern "C"就不行? 那要_stdcall干什么呢?
      

  4.   

    我的意思是难道说, 比如DLL里定义函数名为 A(....)  调用时 GetProcAddress(xx,"A")的话, 就必须要extern "C"了?  GetProcAddress里面A不变的话,没别的办法?
      

  5.   

    extern "C" 告诉编译器编译的时候不要搞名字混成那套东西
    _stdcall 则声明了调用约定。
    有个奇怪的现象就是
    如果你的函数声明里既有 extern "C" 又有 _stdcall
    那么必须在 def 文件中明确导出该函数
    否则用 GetProcAddress 无法找到该函数
      

  6.   

    你也可以在.def中把要用的函数定义,那样就可以找到了。
      

  7.   

    extern "C" 是使用C命名约定(不对函数名称加前后缀)
    __stdcall 是“标准调用”方式,即参数从右往左入栈,由被调用者清栈
    这时候如果不在.def文件中声明,func(int, int)的编译器名称应该是func@8, 
    在.def中导出func时名字这就变成了没有加修饰的func了。
      

  8.   

    其实就是 输出函数名在编译的时候改变的问题。
    声明为extern "C"的意思是,在编译的时候不要修改函数的名字。
    不然在在编译的时候函数的名字有可能会被改变。解决的办法就是 建立一个DEF文件,明确说明输出函数名称,
    这样不用 声明extern "C" 函数的名字也不会被改变了。
      

  9.   

    学习一下,对这几个extern "C" ,_stdcall,_cdecl 的作用不是太清楚。