C#使用不受管理的DLL的应用程序冻结了整个系统



我目前正在Visual Studio 2012中编写一个与RFID卡通信的软件。我在Delphi上写了一个DLL来处理与读卡器的通信。

问题是:我的软件在安装了VS2012的机器上运行良好。在其他系统上,它会冻结自身或整个系统。我在Win XP/7/8上尝试了X32和X64配置。我正在使用.NET 4.0。

连接到读者后,该软件启动了一个背景工作者,该背景工作人员(以200ms的速度)用命令命令读取读者RF场中的库存卡。撞车事故在MUSTH上发生。读者连接后10到20秒。这是代码:

[DllImport("tempConnect.dll", CallingConvention = CallingConvention.StdCall)]
 private static extern int inventory(int maxlen, [In] ref int count, 
                                         IntPtr UIDs, UInt32 HFOffTime);
public String getCardID()
    {
        if (isConnectet())
        {
            IntPtr UIDs = IntPtr.Zero;
            int len = 2 * 8;
            Byte[] zero = new Byte[len];
            UIDs = Marshal.AllocHGlobal(len);
            Thread.Sleep(50);
            Marshal.Copy(zero, 0, UIDs, len);
            int count = 0;
            int erg;
            String ret;
            try
            {
                erg = inventory(len, ref count, UIDs, 50);
            }
            catch (ExternalException) // this doesn't catch anything (iI have set <legacyCorruptedStateExceptionsPolicy enabled="true"/>)
            {
                return "";
            }
            finally
            {
                ret = Marshal.PtrToStringAnsi(UIDs, len);
                IntPtr rslt = LocalFree(UIDs);
                GC.Collect();
            }
            if (erg == 0)
                return ret;
            else
                return zero.ToString();
        }
        else
            return "";
    }

dll是用delphi编写的,代码dll命令是:

function inventory (maxlen: Integer; var count: Integer; 
                  UIDs: PByteArray; HFOffTime: Cardinal = 50): Integer; STDCALL;

我认为某处可能有内存泄漏,但我不知道如何找到它...


编辑:

我在上面的代码中添加了一些想法(explacit gc.collect(),try-catch-fin-fin-fin-fin-fin-fin-fin-fin-fin.

这是代码,它调用getCardid():

每200ms运行一次的动作:

if (!bgw_inventory.IsBusy)
   bgw_inventory.RunWorkerAsync();

ynnc backgroundworker确实:

private void bgw_inventory_DoWork(object sender, DoWorkEventArgs e)
    {
            if (bgw_inventory.CancellationPending)
        {
            e.Cancel = true;
            return;
        }
        else
        {
            String UID = reader.getCardID();
            if (bgw_inventory.CancellationPending)
            {
                e.Cancel = true;
                return;
            }
            if (UID.Length == 16 && UID.IndexOf("") == -1)
            {
                setCardId(UID);
                if (!allCards.ContainsKey(UID))
                {
                    allCards.Add(UID, new Card(UID));
                }
                if (readCardActive || deActivateCardActive || activateCardActive)
                {
                    if (lastActionCard != UID)
                        actionCard = UID;
                    else
                        setWorkingStatus("OK", Color.FromArgb(203, 218, 138));
                }
            }
            else
            {
                setCardId("none");
                if (readCardActive || deActivateCardActive || activateCardActive)
                    setWorkingStatus("waiting for next card", Color.Yellow);
            }
        }
   }

编辑

到现在为止,我已经在代码上进行了一些小返工(上面的更新)。现在只有应用。在" tempconnect.dll"上使用0xc00000fd(堆栈溢出)崩溃。这不会在安装VS2012的系统上发生,或者如果我将DLL与本机Delphi!一起使用有人还有其他想法吗?


编辑

现在,我将DLL记录起来是堆叠的,发现了一些奇怪的东西:如果是从我的C#程序中调用并进行了调查,那么堆叠量正在不断上下改变。如果我从天然deplhi程序中做同样的事情,那么堆叠是恒定的!因此,我将进行进一步的调查,但是我不知道,我必须搜索什么...

我有点担心如何使用该Marshal对象。当您担心内存泄漏时,它似乎是经常分配记忆,但我认为它从未明确发布。垃圾收集器应该(手术单词)正在照顾好这一点,但是您说自己的代码中有一些不受管理的代码。张贴的信息很难判断非管理代码从哪里开始。

查看此问题,以获取一些良好的技术来查找.NET本身中的内存泄漏 - 这将为您提供大量信息,以了解代码的托管端如何使用内存(即,您可以直接可以直接控制)。使用带有断点的Windows Performance监视器来关注系统的整体健康状况。如果.NET似乎表现出色,但是WPM显示出一些尖锐的尖峰,则可能是在不受管理的代码中。除了您的使用情况外,您无法真正控制任何东西,因此可能是时候回到文档了。

最新更新