【发布时间】:2011-05-28 05:29:17
【问题描述】:
在一个使用 server.dll 和 client.exe 的项目中,我已经 dllexported 来自服务器 dll 的服务器符号,而 没有 dllimported 将它放入客户端 exe。
仍然,应用程序链接并启动,没有任何问题。是不是不需要dllimport,那么???
详情:
我有这个“服务器”dll:
// server.h
#ifdef SERVER_EXPORTS
#define SERVER_API __declspec(dllexport)
#else
#define SERVER_API // =====> not using dllimport!
#endif
class SERVER_API CServer {
static long s;
public:
CServer();
};
// server.cpp
CServer::CServer(){}
long CServer::s;
还有这个客户端可执行文件:
#include <server.h>
int main() {
CServer s;
}
服务器命令行:
cl.exe /Od /D "WIN32" /D "_DEBUG" /D "_WINDOWS" /D "_USRDLL"
/D "SERVER_EXPORTS" /D "_UNICODE" /D "UNICODE" /D "_WINDLL"
/Gm /EHsc /RTC1 /MDd /Yu"stdafx.h"
/Fp"Debug\server.pch" /Fo"Debug\\" /Fd"Debug\vc80.pdb"
/W3 /nologo /c /Wp64 /ZI /TP /errorReport:prompt
cl.exe /OUT:"U:\libs\Debug\server.dll" /INCREMENTAL:NO /NOLOGO /DLL
/MANIFEST /MANIFESTFILE:"Debug\server.dll.intermediate.manifest"
/DEBUG /PDB:"u:\libs\Debug\server.pdb"
/SUBSYSTEM:WINDOWS /MACHINE:X86 /ERRORREPORT:PROMPT
kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib
shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib
客户端命令行:
cl.exe /Od /I "..\server"
/D "WIN32" /D "_DEBUG" /D "_CONSOLE" /D "_UNICODE" /D "UNICODE"
/Gm /EHsc /RTC1 /MDd /Fo"Debug\\" /Fd"Debug\vc80.pdb" /W3 /c /Wp64 /ZI /TP
.\client.cpp
cl.exe /OUT:"U:\libs\Debug\Debug\client.exe" /INCREMENTAL
/LIBPATH:"U:\libs\Debug"
/MANIFEST /MANIFESTFILE:"Debug\client.exe.intermediate.manifest"
/DEBUG /PDB:"u:\libs\debug\debug\client.pdb"
/SUBSYSTEM:CONSOLE /MACHINE:X86
server.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib
advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib
【问题讨论】:
-
其实是个好问题。 MSDN Docs --- msdn.microsoft.com/en-us/library/3y1sfaz2(VS.80).aspx --- 并没有告诉我使用
extern(具有正确的调用约定和名称修改)和指定导入库是否有任何好处。 -
__declspec(dllexport)对类和类成员非常、非常 脆弱。单独的 server.dll 的目的是什么?当与/delayload:server.dll配对时,类上唯一的__declspec(dllexport)做得好的就是减少进程启动I/O。任何其他感知到的优势(例如,想象能够在不重新编译应用程序的情况下修补 DLL 逻辑)实际上违反了单一定义规则并且不可靠。 -
@xtofl:解耦很好,源码其实也可以解耦。但是类和类成员上的
__declspec(dllexport)会导致二进制文件 紧密耦合。换句话说,通过使用静态库并且在任何地方都没有__declspec(dllexport),您将拥有相同级别的耦合,并且部署负担要少得多。 -
@Sergey:最好将所有代码都放在一个 .DLL 模块中,只为 7 个应用程序(甚至参数)中的每一个导出与
main函数相对应的纯 C 函数到单个 .exe,ala busybox)。至少在 Windows 上,这是我们在这里讨论的平台,Qt 不提供二进制兼容性。您必须使用应用程序使用的特定编译器和命令行选项自己构建 Qt 库,以避免违反 ODR。在这种情况下,您不妨再次使用静态库。 -
@marshalcraft 这不是博客——把你的咆哮留到别处吧。请:)。
标签: c++ dllimport visual-c++-2005 declspec