C ++ / CLI应用程序在发布版本上随机崩溃

时间:2011-01-09 20:53:12

标签: exception dll constructor random c++-cli

我创建了一个C ++ / CLI混合DLL,我在C#Winforms应用程序中使用它。 我仔细检查了Build config,以确保我链接到Debug模式下的调试库和Release中的非调试库。

现在应用程序什么都不做,只是在这样的托管包装器中创建一个本机类(单例模式以确保该类的单个实例):

static ManagedClassWrapper ^ GetInstance(){
                if(_me == nullptr){
                    _me = gcnew ManagedClassWrapper();
                    _me->_Impl = new NativeClass();
                }

                return _me;
            };

其中_me和_impl是

private:
    NativeClass * _Impl;
    static ManagedClassWrapper ^ _me = nullptr;

在点击按钮的表单中我执行此操作:

private void button1_Click(object sender, EventArgs e)
{
    ManagedClassWrapper mcw = ManagedClassWrapper.GetInstance();
}

我也像往常一样有一个标准的原生入口点DllMain。在DEBUG构建中我使用

_CrtSetReportHook( QaDMemManager::report );
_CrtSetDbgFlag((_CRTDBG_LEAK_CHECK_DF) | _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG));

在DllMain的开头,在DEBUG构建中我也重新定义了新的:

#ifdef _DEBUG
#define _CRTDBG_MAP_ALLOC
#define LOG_LEVEL Logger::NOTICE
#include <stdlib.h>
#include <crtdbg.h>
#pragma warning(disable:4291)
#define new new(_NORMAL_BLOCK,__FILE__, __LINE__)
#else
#define LOG_LEVEL Logger::INFO
#endif

正如我通常为我的非MFC应用程序所做的那样,以获得良好的内存泄漏。

NativeClass的构造函数为空。

在Debug版本中,一切正常,我看到本机代码中存在内存泄漏,没有崩溃。

但是在Release版本中,我的应用程序中只有一次崩溃,当我点击该按钮1时崩溃。 这意味着:我可以启动我的应用程序的10个实例,9无论我点击button1多少次都可以正常工作,但每次点击button1时第10个都会崩溃(崩溃后我在异常窗口中点击继续等等我可以多次点击button1。

例外情况如下:

************** Exception Text **************
System.TypeInitializationException: The type initializer for '<Module>' threw an exception. ---> System.AccessViolationException: Attempted to read or write protected memory. This is often an indication that other memory is corrupt.
   at _initterm((fnptr)* pfbegin, (fnptr)* pfend)
   at <CrtImplementationDetails>.LanguageSupport.InitializeNative(LanguageSupport* )
   at <CrtImplementationDetails>.LanguageSupport._Initialize(LanguageSupport* )
   at <CrtImplementationDetails>.LanguageSupport.Initialize(LanguageSupport* )
   at .cctor()
   --- End of inner exception stack trace ---
   at TestAudioInOut.TestForm.button1_Click(Object sender, EventArgs e)
   at System.Windows.Forms.Control.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnClick(EventArgs e)
   at System.Windows.Forms.Button.OnMouseUp(MouseEventArgs mevent)
   at System.Windows.Forms.Control.WmMouseUp(Message& m, MouseButtons button, Int32 clicks)
   at System.Windows.Forms.Control.WndProc(Message& m)
   at System.Windows.Forms.ButtonBase.WndProc(Message& m)
   at System.Windows.Forms.Button.WndProc(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
   at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
   at System.Windows.Forms.NativeWindow.Callback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)


************** Loaded Assemblies **************
mscorlib
    Assembly Version: 4.0.0.0
    Win32 Version: 4.0.30319.1 (RTMRel.030319-0100)
    CodeBase: file:///C:/Windows/Microsoft.NET/Framework/v4.0.30319/mscorlib.dll
----------------------------------------
TestAudioInOut
    Assembly Version: 1.0.0.0
    Win32 Version: 1.0.0.0
    CodeBase: file:///V:/Test/bin/Release/Test.exe
----------------------------------------
System.Windows.Forms
    Assembly Version: 4.0.0.0
    Win32 Version: 4.0.30319.1 built by: RTMRel
    CodeBase: file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Windows.Forms/v4.0_4.0.0.0__b77a5c561934e089/System.Windows.Forms.dll
----------------------------------------
System.Drawing
    Assembly Version: 4.0.0.0
    Win32 Version: 4.0.30319.1 built by: RTMRel
    CodeBase: file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Drawing/v4.0_4.0.0.0__b03f5f7f11d50a3a/System.Drawing.dll
----------------------------------------
System
    Assembly Version: 4.0.0.0
    Win32 Version: 4.0.30319.1 built by: RTMRel
    CodeBase: file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System/v4.0_4.0.0.0__b77a5c561934e089/System.dll
----------------------------------------
Mixed.DLL
    Assembly Version: 1.0.4026.39493
    Win32 Version: 
    CodeBase: file:///V:/Test/bin/Release/Mixed.DLL
----------------------------------------

可能是什么问题(据我所知,TypeInitializationException意味着构造对象有问题)以及为什么它只处于Release模式?

1 个答案:

答案 0 :(得分:5)

它与你发布的代码片段没有任何关系,在创建ManagedClassWrapper之前代码炸弹。 <Module>类是围绕您编写的所有非ref类代码的包装类。它试图调用非托管代码的初始化程序时会发生炸弹。这是一种AccessViolation,是非托管代码通常采用的方式。

要调试它,您必须在C#项目中启用非托管调试。 Project + Properties,Debug选项卡,勾选“启用非托管代码调试”。然后调试+异常,勾选“Win32 Exceptions”上的Thrown标志。运行代码直到崩溃发生,调试器将停在崩溃位置。应该让你知道bug的位置。在使用非托管代码时,请使用您熟悉的常规调试技术。祝你好运。