Perchè / quando è __declspec (dllimport) non necessaria?
-
11-10-2019 - |
Domanda
In un progetto che utilizza un server.dll e Client.exe, ho dllexport
ed un simbolo server dal dll server e non dllimport
ed in exe client.
Ancora, i collegamenti delle applicazioni e partenze, senza alcun problema. È dllimport
non necessario, quindi ???
dettagli:
Ho questa dll 'server':
// 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;
e questo eseguibile client:
#include <server.h>
int main() {
CServer s;
}
La riga di comando del server:
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
riga di comando client:
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
Soluzione
Non è necessario. Si tratta di un'ottimizzazione, un suggerimento per il compilatore che la DLL sta per esportare il puntatore a funzione direttamente piuttosto che solo una voce nel IAT della DLL. Il puntatore funzione esportata per una funzione denominata foo () sarà __imp_foo. Che le permette di generare codice migliore, il salvataggio di un carico di puntatore a funzione dal IAT e di un salto indiretto. Si tratta di un'ottimizzazione dei tempi, non lo spazio.
post sul blog ha la i dettagli.