我有一个名为tccdvc.dll的DLL,它是SDK的一部分:
DLL是用C++编写的,检查DLL显示它与链接器版本6.0链接,因此我假设它是用VC++ 6.0编写的.DLL没有源代码,只有.lib文件和.h文件.所有导出的函数都声明为extern"C"(因此没有C++名称修改)和APIENTRY(所以__stdcall).
我在Windows XP SP3(32位)上的Visual Studio 2010中编写了一个C++(而不是.NET)程序来访问此tccdvc.dll.这在使用提供的.lib文件和使用LoadLibrary/GetProcAddress时都可以正常工作.我还编写了一个C++ DLL(让我们称之为mywrapper.dll),它使用tccdvc.dll,并且在两个版本中,一个使用.lib文件,另一个使用LoadLibrary/GetProcAddress.再次,这很好.此mywrapper.dll使用__cdecl调用约定.它包含一个名为InitMyWrapperDLL()的函数,它加载tccdvc.dll.使用LoadLibrary/GetProcAddress的mywrapper.dll版本的代码如下:
typedef int (APIENTRY *TCCPROCTYPE01)();
HMODULE TCCmodule;
TCCPROCTYPE01 Proc_TCC_DVCOpen;
extern "C" __declspec(dllexport) void InitMyWrapperDLL ()
{ TCCmodule = LoadLibrary("tccdvc.dll");
Proc_TCC_DVCOpen = (TCCPROCTYPE01)GetProcAddress(TCCmodule, "TCC_DVCOpen");
...
}
Run Code Online (Sandbox Code Playgroud)
再次,使用C++前端,这很好.但是,当从C#(在同一台机器上)调用它时,LoadLibrary("tccdvc.dll")调用返回NULL.在C#中,我正在使用:
[DllImport("mywrapper.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi, ExactSpelling=true, EntryPoint="InitMyWrapperDLL")]
private static extern void InitMyWrapperDLL ();
...
InitMyWrapperDLL();
Run Code Online (Sandbox Code Playgroud)
当使用提供的tccdvc.lib文件编译mywrapper.dll时,它也会失败,错误代码为0x8007045a(也称为1114),这意味着DLL初始化失败,并且它将mywrapper.dll作为DLL的名称.事实证明,失败是因为tccdvc.dll,它通过mywrapper.dll加载.
在C#中使用以下内容也会失败:
[DllImport("tcc.dll", CallingConvention=CallingConvention.StdCall, CharSet=CharSet.Ansi, ExactSpelling=true, EntryPoint="TCC_DVCOpen")]
private static extern Int32 TCC_DVCOpen ();
...
TCC_DVCOpen();
Run Code Online (Sandbox Code Playgroud)
我在声明中也使用了"不安全",但这没有任何区别.可预测,因为LoadLibrary()失败,所以它甚至没有到达TCC_DVCOpen().
为了查明问题,我再次使用mywrapper.dll的LoadLibrary/GetProcAddress版本并将以下代码放入我的C#程序中:
[DllImport("kernel32.dll")]
private static extern IntPtr LoadLibrary (string lpLibFileName);
[DllImport("kernel32.dll")]
private static extern Int32 GetLastError (); …Run Code Online (Sandbox Code Playgroud) 考虑以下代码:
int i, k, m;
k = 12;
m = 34;
for (i = 0; i < 2; i++) ((i & 1) ? k : m) = 99 - i;
printf("k: %ld m: %ld\n\n", k, m);
Run Code Online (Sandbox Code Playgroud)
在这个愚蠢的例子中,条件运算符表达式是一个快捷方式:
if (i & 1) k = 99 - i; else m = 99 - i;
Run Code Online (Sandbox Code Playgroud)
我的编译器没有抱怨并执行这段代码给出了预期的输出
k: 98 m: 99
Run Code Online (Sandbox Code Playgroud)
不过,我的问题是,如果这是符合C标准的有效代码吗?我以前从未见过这样的东西.