加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 综合聚焦 > 服务器 > Windows > 正文

带有非托管DLL的C#应用??程序冻结了整个系统

发布时间:2020-12-14 02:45:29 所属栏目:Windows 来源:网络整理
导读:我目前正在Visual Studio 2012中编写一个用于与RFID卡通信的软件. 我有一个用Delphi编写的DLL来处理与读卡器的通信. 问题是:我的软件在安装了VS2012的机器上运行良好.在其他系统上,它会冻结自身或整个系统. 我在Win XP / 7/8上尝试了x32和x64配置. 我正在使
我目前正在Visual Studio 2012中编写一个用于与RFID卡通信的软件.
我有一个用Delphi编写的DLL来处理与读卡器的通信.

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

在连接到阅读器之后,软件启动backgroundWorker,该阅读器以读取器RF场中的库存卡的命令轮询(以200ms的速率)阅读器.崩溃通常发生在ca.读卡器连接后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,UIDs,len);
            int count = 0;
            int erg;
            String ret;
            try
            {
                erg = inventory(len,ref count,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;

我想某处可能存在内存泄漏,但我不知道如何找到它…

编辑:

我在上面的代码中添加了一些想法(显式GC.Collect(),try-catch-finally),但它仍然无效.

这是调用getCardID()的代码:

该动作每200ms运行一次:

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

异步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);
            }
        }
   }

编辑

到目前为止,我已经在代码中进行了一些小的reworks(上面的更新).现在只有App.在“tempConnect.dll”崩溃时出现0xC00000FD(堆栈溢出).安装VS2012的系统或我使用本机Delphi的DLL时不会发生这种情况!
有没有人有任何其他想法?

编辑

现在我让DLL记录它的stacksize并发现一些奇怪的东西:
如果它从我的C#Programm中调用和轮询,则stacksize会不断地上下变化.
如果我从自然的Deplhi程序中做同样的事情,stacksize是不变的!
所以我会做进一步的调查,但我不知道,我要搜索什么……

解决方法

我有点担心如何使用Marshal对象.当您担心内存泄漏时,它似乎经常分配内存,但我没有看到它明确地释放它.垃圾收集器应该(操作词)处理它,但你说你自己在混合中有一些非托管代码.发布的信息很难说明非托管代码的开始位置.

查看this question是否有一些很好的技术来查找.NET本身的内存泄漏 – 这将为您提供大量有关如何在代码的托管端使用内存的信息(即,您可以直接控制的部分).使用带有断点的Windows性能监视器来监视系统的整体运行状况.如果.NET似乎表现不佳,但是WPM显示出一些尖锐的峰值,那么它可能在非托管代码中.除了你的使用之外,你无法控制任何东西,所以可能是时候回到文档了.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读