LNK2019:win32下编写DLL,应用程序找不到DLL接口函数的一种情况
2016-06-14 14:25
417 查看
最近项目技术研究结束,开始代码封装,产品最终是以动态库的形式发布,于是开始整理代码,把原来以静态库方式编译的函数库改为动态库编译,动态库编译正常。
然后用编译测试程序(调用动态库的函数)时报错了,
2>test_detect.obj : error LNK2019: 无法解析的外部符号 “_declspec(dllimport) void __cdecl gdface::set_global_run_config(bool,bool)” (__imp?set_global_run_config@gdface@@YAX_N0@Z),该符号在函数 main 中被引用
令我感到奇怪的是,动态库导出的函数有好几个,只有这个函数报错,我打开生成的动态库导入文件(.lib)。通过字符串查找,的确找不到set_global_run_config函数。这证明是在编译生成动态库的时候,就出了问题。
但为什么偏偏只有这一个函数出问题呢?我仔细比较这个函数和其他函数的区别,发现那些能正常导出符号的函数在同一个cpp文件A中,只有这个问题函数在另一个cpp文件B中。
而A文件有include下面的接口头文件,B文件并没有include这个接口头文件
detect_cl.h
于是我在B文件开头加入了#include “detect_cl.h”,编译正常。
结论就是:
对于动态库接口函数,在编译期间就与普通函数有区别,而不只是在连接期间,所以函数所在的cpp/c文件必须#include这个接口函数的定义头文件。
然后用编译测试程序(调用动态库的函数)时报错了,
2>test_detect.obj : error LNK2019: 无法解析的外部符号 “_declspec(dllimport) void __cdecl gdface::set_global_run_config(bool,bool)” (__imp?set_global_run_config@gdface@@YAX_N0@Z),该符号在函数 main 中被引用
令我感到奇怪的是,动态库导出的函数有好几个,只有这个函数报错,我打开生成的动态库导入文件(.lib)。通过字符串查找,的确找不到set_global_run_config函数。这证明是在编译生成动态库的时候,就出了问题。
但为什么偏偏只有这一个函数出问题呢?我仔细比较这个函数和其他函数的区别,发现那些能正常导出符号的函数在同一个cpp文件A中,只有这个问题函数在另一个cpp文件B中。
而A文件有include下面的接口头文件,B文件并没有include这个接口头文件
detect_cl.h
#ifndef FACEDETECT_DETECT_CL_H_ #define FACEDETECT_DETECT_CL_H_ #include <vector> // 动态库导入导出声明宏定义 #ifdef _WIN32 #ifdef DETECT_CL #define DETECT_CL_DLL_DECL __declspec(dllexport) #define DETECT_CL_EXTERN #else #define DETECT_CL_DLL_DECL __declspec(dllimport) #define DETECT_CL_EXTERN #endif #else #define DETECT_CL_DLL_DECL #define DETECT_CL_EXTERN #endif namespace gdface{ DETECT_CL_DLL_DECL gf_rect_def gf_rect_unnormalization( const gf_rect_norm &norm_obj,const gf_int2 &img_size); DETECT_CL_DLL_DECL object_info_vector& convert_def(object_info_vector& objects, const gf_int2 &img_size); DETECT_CL_DLL_DECL object_info_vector convert_def(const object_info_vector& objects, const gf_int2 &img_size); DETECT_CL_DLL_DECL void set_global_run_config(bool debug_show,bool kerne_message); }/*namespace gdface*/ #endif /* FACEDETECT_DETECT_CL_H_ */
于是我在B文件开头加入了#include “detect_cl.h”,编译正常。
结论就是:
对于动态库接口函数,在编译期间就与普通函数有区别,而不只是在连接期间,所以函数所在的cpp/c文件必须#include这个接口函数的定义头文件。
相关文章推荐
- FREEBASIC 编译可被python调用的dll函数示例
- DLL(Dynamic Linkable Library) 详解说明
- delphi中一个值得大家来考虑的DLL问题
- 将ocx文件转换成C#程序引用的DLL文件的办法
- win32下进程间通信(共享内存)实例分析
- C++与C#互调dll的实现步骤
- C++调用C#的DLL实现方法
- 自己写的文件操作的function和Sub vb.net dll
- C# 调用C++写的dll的实现方法
- C#使用DllImport调用非托管的代码的方法
- 使用 JScript 创建 .exe 或 .dll 文件的方法
- 深入解析JVM对dll文件和对类的装载过程
- c#中xml文档注释编译dll引用到其它项目示例
- WMI中的Win32_PingStatus类(ping命令实现)
- win32安装配置非安装版的MySQL
- 一个win32窗口创建示例
- .dll 文件反编译的工具软件集合
- C#调用C++版本dll时的类型转换需要注意的问题小结
- java使用JNA(Java Native Access)调用dll的方法