在 C++(win32 应用程序)中使用 C# 类时出现 EEFileLoadException
Posted
技术标签:
【中文标题】在 C++(win32 应用程序)中使用 C# 类时出现 EEFileLoadException【英文标题】:EEFileLoadException when using C# classes in C++(win32 app) 【发布时间】:2008-09-18 15:49:31 【问题描述】:出于部署原因,我尝试使用 IJW 在 C++ 中包装 C# 程序集,而不是使用 COM 可调用包装器。
我在其他项目上也做过,但在这个项目上,我得到了一个 EEFileLoadException。任何帮助将不胜感激!
托管 C++ 包装代码(在 DLL 中):
extern "C" __declspec(dllexport) IMyObject* CreateMyObject(void)
//this class references c# in the constructor
return new CMyWrapper( );
extern "C" __declspec(dllexport) void DeleteMyObject(IMyObject* pConfigFile)
delete pConfigFile;
extern "C" __declspec(dllexport) void TestFunction(void)
::MessageBox(NULL, _T("My Message Box"), _T("Test"), MB_OK);
测试代码(这是一个EXE):
typedef void* (*CreateObjectPtr)();
typedef void (*TestFunctionPtr)();
int _tmain testwrapper(int argc, TCHAR* argv[], TCHAR* envp[])
HMODULE hModule = ::LoadLibrary(_T("MyWrapper"));
_ASSERT(hModule != NULL);
PVOID pFunc1 = ::GetProcAddress(hModule, "TestFunction");
_ASSERT(pFunc1 != NULL);
TestFunctionPtr pTest = (TestFunctionPtr)pFunc1;
PVOID pFunc2 = ::GetProcAddress(hModule, "CreateMyObject");
_ASSERT(pFunc2 != NULL);
CreateObjectPtr pCreateObjectFunc = (CreateObjectPtr)pFunc2;
(*pTest)(); //this successfully pops up a message box
(*pCreateObjectFunc)(); //this tosses an EEFileLoadException
return 0;
对于它的价值,事件日志报告以下内容: .NET 运行时版本 2.0.50727.143 - 致命的执行引擎错误 (79F97075) (80131506)
很遗憾,Microsoft 没有关于该错误的信息。
【问题讨论】:
您是否进行了测试来验证程序集是否从同一目录加载? 当我手动将托管 DLL 复制到 exe 的目录中时,它可以在没有这些额外代码的情况下工作。 【参考方案1】:问题在于 DLL 的位置。
c:\dlls\managed.dll c:\dlls\wrapper.dll c:\exe\my.exe我通过将 managed.dll 复制到 c:\exe 中确认了这一点,它可以正常工作。显然,CLR 不会在非托管 DLL 的路径中查找托管 DLL,而只会在可执行文件所在的位置查找它。 (或在 GAC 中)。
出于不值得讨论的原因,这是我需要的结构,这意味着我需要让 CLR 帮助定位托管 dll。见以下代码:
AssemblyResolver.h:
/// <summary>
/// Summary for AssemblyResolver
/// </summary>
public ref class AssemblyResolver
public:
static Assembly^ MyResolveEventHandler( Object^ sender, ResolveEventArgs^ args )
Console::WriteLine( "Resolving..." );
Assembly^ thisAssembly = Assembly::GetExecutingAssembly();
String^ thisPath = thisAssembly->Location;
String^ directory = Path::GetDirectoryName(thisPath);
String^ pathToManagedAssembly = Path::Combine(directory, "managed.dll");
Assembly^ newAssembly = Assembly::LoadFile(pathToManagedAssembly);
return newAssembly;
;
包装器.cpp:
#include "AssemblyResolver.h"
extern "C" __declspec(dllexport) IMyObject* CreateMyObject(void)
try
AppDomain^ currentDomain = AppDomain::CurrentDomain;
currentDomain->AssemblyResolve += gcnew ResolveEventHandler( AssemblyResolver::MyResolveEventHandler );
return new CMyWrapper( );
catch(System::Exception^ e)
System::Console::WriteLine(e->Message);
return NULL;
【讨论】:
我花了好几个小时试图弄清楚我的混合 DLL 出了什么问题,结果却发现它抛出了一个EEFileLoadException
。谢谢你,它帮助了:)【参考方案2】:
第一个问题是确保调试器类型设置为混合。然后你会得到有用的异常。
【讨论】:
我遇到了他用托管 C++ 编写的 Excel 插件的问题,实际上我发现在混合调试中我得到 less 有用的异常;特别是我没有看到 EEFileLoadException 本身,只是随后的几次重新抛出。 这个建议可能为我节省了一个小时的调试时间。【参考方案3】:对于使用混合模式 dll(您的 EXE)的本机应用程序,请将 **“调试器类型”更改为“混合”模式。 (进入项目属性 -> 配置属性 -> 调试)
还有一些其他点(可能与您无关),但根据我的经验,它们可能会导致问题。 - 在 Windows 8(具有更严格的安全性)上,请尝试以管理员身份启动您的 VS。 - 确保对于 x86 配置,您使用的是 x86 二进制文件。 - 注意 StrongName 验证,如果您在托管 C++ 中使用的 C# 程序集已签名,请考虑对混合模式 dll 进行签名。
希望这会有所帮助。
【讨论】:
这为我解决了问题!现在我可以看到实际的错误...谢谢!【参考方案4】:如果其他人偶然发现了这个问题,并且您使用的是动态程序集名称:请确保您剥离了程序集名称,它可能包含您可能不会使用的版本、文化和其他内容。
即,您的 MyResolveEventHandler 应采用以下形式:
static Assembly^ MyResolveEventHandler( Object^ sender, ResolveEventArgs^ args )
Console::WriteLine( "Resolving..." );
String^ assemblyName = args->Name;
// Strip irrelevant information, such as assembly, version etc.
// Example: "Acme.Foobar, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null"
if( assemblyName->Contains(",") )
assemblyName = assemblyName->Substring(0, assemblyName->IndexOf(","));
Assembly^ thisAssembly = Assembly::GetExecutingAssembly();
String^ thisPath = thisAssembly->Location;
String^ directory = Path::GetDirectoryName(thisPath);
String^ pathToManagedAssembly = Path::Combine(directory, assemblyName );
Assembly^ newAssembly = Assembly::LoadFile(pathToManagedAssembly);
return newAssembly;
【讨论】:
【参考方案5】:在调试 ASP.NET MVC 应用程序期间,iisexpress.exe 经常抛出 C++ EEFileLoadException。调用堆栈和 C++ 异常本身对于帮助我解决问题并没有太大帮助。
在直接查看 C++ 异常中给出的指针地址后,我最终发现了一个库字符串,它指向不再使用的旧版本。这反过来又是由于我的 web.config 文件中的一个过期条目:
<runtime>
<assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
<dependentAssembly>
<assemblyIdentity name="Microsoft.Owin.Security.OAuth" publicKeyToken="31bf3856ad364e35" />
<bindingRedirect oldVersion="0.0.0.0-3.0.1.0" newVersion="3.0.1.0" />
</dependentAssembly> </assemblyBinding> </runtime>
我已通过 NuGet 将各种 Microsoft.Own 安全库升级到版本 4.0.30319,但配置中的这一行指示服务器将调用重定向到版本 3.0.1.0,现在不再是我项目的一部分。更新配置解决了我的问题。
【讨论】:
【参考方案6】:当您在使用 C++ 托管 dll 的调试器 C++ 本机项目中运行时,您可能会遇到此异常。当 VS2010 捕获它并且您的应用程序在某些链异常之后将被中止时,您可以尝试在异常过滤器(菜单|调试|异常)中禁用所有 C++ 异常。您仍会在输出中看到此异常,但您的应用程序不会中止
【讨论】:
-1:这似乎是一个不解决问题的坏建议以上是关于在 C++(win32 应用程序)中使用 C# 类时出现 EEFileLoadException的主要内容,如果未能解决你的问题,请参考以下文章
UWP C# 管道客户端无法连接到 Win32 C++ 管道服务器
C++ Win32 中的 C# DateTime.ToUniversalTime() [关闭]
将 C++ Win32 控制台项目类集成到 Visual Studio 2008 中的 Visual C++(Windows 窗体应用程序)项目中
如何在 C++ 中使用 Win32_PerfRawData_Tcpip_NetworkInterface 类获取当前带宽