我目前正在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, 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 "\0";
}
finally
{
ret = Marshal.PtrToStringAnsi(UIDs, len);
IntPtr rslt = LocalFree(UIDs);
GC.Collect();
}
if (erg == 0)
return ret;
else
return zero.ToString();
}
else
return "\0";
}
DLL是用Delphi编写的,代码DLL命令是:
function inventory (maxlen: Integer; var count: Integer;
UIDs: PByteArray; HFOffTime: Cardinal = 50): Integer; STDCALL;
我认为某处可能存在内存泄漏,但我不知道如何找到它......
编辑:
我在上面的代码中添加了一些想法(显式GC.Collect(),try-catch-finally),但它仍然无效。
以下是调用getCardID()的代码:
每200毫秒运行一次的动作:
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("\0") == -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的系统上或者我将DLL与本机Delphi一起使用时,不会发生这种情况! 有人有任何其他想法吗?
修改
现在我让DLL记录它的stacksize并发现一些奇怪的东西: 如果它从我的C#Programm中调用和轮询,则stacksize会不断地上下变化。 如果我从自然的Deplhi程序中做同样的事情,stacksize是不变的! 所以我会做进一步的调查,但我不知道,我要搜索什么......
答案 0 :(得分:1)
我有点担心如何使用Marshal
对象。当您担心内存泄漏时,它似乎经常分配内存,但我没有看到它明确地释放它。垃圾收集器应该(操作词)处理它,但你说你自己在混合中有一些非托管代码。发布的信息很难说明非托管代码的开始位置。
查看this question有关在.NET本身中查找内存泄漏的一些好方法 - 这将为您提供大量有关如何在代码的托管端使用内存的信息(即,部分你可以直接控制)。使用带有断点的Windows性能监视器来监视系统的整体运行状况。如果.NET似乎表现不佳,但是WPM显示出一些尖锐的峰值,那么它可能在非托管代码中。除了你在那里的使用之外,你无法控制任何东西,因此可能是时候回到文档了。