我会尝试尽可能详细地描述我的问题,但如果需要更详细的解释,请告诉我。
为了简化,假设我有 3 个 DLL(实际上我有更多,但我猜这不是很重要):
managed-1.dll
- 托管 DLL(用 C# .NET 4.0 编写) - 处理请求,并在某些请求期间调用第二个 DLL 中的非托管代码unmanagement.dll
- 非托管 DLL(用老式 VC++ 6.0 编写) - 执行多个操作,有时调用第三个 DLLmanaged-2.dll
- 托管 DLL(用 CLI/C++ .NET 3.5 编写) - 我的问题的根源
我在 3 种不同的场景中运行我的代码:
我从控制台应用程序调用
managed-1.dll
- 一切正常我从
ASP.NET Development Server
调用managed-1.dll
- 一切都运行良好我从
IIS
调用managed-1.dll
- 一切正常,直到整个序列managed-1.dll -> unmanaged.dll -> 涉及 Managed-2.dll
。
在场景 3 中,抛出 StackOverflowException。调试器显示不涉及递归。另外很明显,该异常发生在以下类型的调用堆栈中:
managed-1.dll::CallUnmanagementCode()
unmanagement.dll::SomeMethod1()
unmanagement.dll::SomeMethod2()
unmanagement.dll::CallManagedCode()
managed-2.dll::CallUnmanagementCode()
!!用__declspec(dllexport)
标记,并且不使用任何托管类型!!managed-2.dll::FailingMethod()
!!使用托管类型;一开始(连第一行代码都没有执行)就发生了异常!!
更有趣的事情是:调试器在 FailingMethod
中显示的参数值与方法调用点中的值不同。
如果有人有任何线索,请提供建议。
解决方案:问题与托管-非托管内容无关,而是与 IIS stack size 有关。 。对我来说 editbin 的用法工具不是可接受的解决方案。所以我的解决方案 - 在调用 unmanagement.dll
之前创建新线程并将堆栈设置为 1 MB:
var result = unchecked ((int)0x800000FF);
var thread = new Thread(() => { result = pinvoke_func(); }, 1024 * 1024); // 1MB
thread.Start();
thread.Join();
最佳答案
普通 Windows 应用程序的堆栈大小显然为 1MB,但对于 IIS,堆栈大小为 256KB(请参阅 http://support.microsoft.com/kb/932909 )。这或许可以解释为什么应用程序在控制台应用程序中运行,但在 IIS 中却不然。
我不知道 ASP.NET 开发服务器的堆栈大小到底是多少,但可能是 1MB,请参阅此问题以获取更多信息:stackoverflowexception-in-iis7-but-not-in-cassini
显然可以使用 editbin 增加 IIS 的堆栈大小。
有关说明,请参阅本文:Stack sizes in IIS - affects ASP.NET
关于.net - IIS 托管 -> 非托管 -> 托管 -> StackOverflowException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11628627/