- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
所以这是一个相当小的问题,但有很大的解释。正如标题所指出的,我收到一个未处理的异常,告诉我我的安全句柄已关闭。我可能需要做的是用越来越多的代码编辑这篇文章几次,以帮助我诊断问题所在。
我正在使用 POS for .NET 为我的 RFID 和 MSR 设备制作服务对象。尽管我的设备相同,但我有 2 个不同的虚拟 COM 端口芯片可以与这些设备通信。一个由 Silicon labs 提供,另一个由 FTDI 提供。我想使用 POS for .NET 的即插即用功能,所以我给了它我的硬件 ID。因为它是即插即用的,所以我可以使用完整的硬件路径,然后我可以通过调用 PInvoke 创建一个 SafeFileHandle,并使用该 SafeFileHandle 创建一个 FileStream。 FTDI 芯片不允许我像那样直接与设备对话,所以我必须获取设备的友好名称,然后使用互斥锁拉出 COM 端口,然后创建一个 SerialPort 实例。该步骤效果很好。仅供引用,我尝试使用两个芯片的友好名称来获取 COM 端口,而 Silicon Labs 的一个(由于某些奇怪的原因)没有使用端口 GUID 使用 SetupAPI.GetDeviceDetails 列出。我不确定那个,因为在设备管理器中,Silicon labs 设备类 Guid 是端口 GUID。
既然 SerialPort 和 FileStream 都有一个 Stream 对象,我决定使用它来读取和写入该端口。问题在于,如果我向 MSR 设备发送 RFID 命令,MSR 设备不会以任何方式响应。因此,如果我使用此代码 int fromReader = ReaderStream.ReadByte();
,我的线程将被阻塞。这是一个阻塞调用,需要至少 1 个字节才能继续。所以我环顾四周,似乎唯一的解决方案是使用单独的线程并设置超时。如果发生超时,则中止线程。
Thread t = new Thread(new ThreadStart(ReadFromStream));
t.Start();
if (!t.Join(timeout))
{
t.Abort();
}
(t.Abort 已经被 try/catch 包围了,但没有用,因为它没有解决我删除它的问题)
ReadFromStream 是 RFID Device 中的抽象方法。这是实现之一
protected override void ReadFromStream()
{
var commandLength = USN3170Constants.MIN_RESPONSE_LENGTH;
var response = new System.Collections.Generic.List<byte>(USN3170Constants.MIN_RESPONSE_LENGTH);
for (int i = 0; i <= commandLength; i++)
{
int fromReader = ReaderStream.ReadByte();
if (fromReader == -1) break; //at end of stream
response.Add((byte)fromReader);
if (response.Count > USN3170Constants.DATA_LENGTH_INDEX && response[USN3170Constants.DATA_LENGTH_INDEX] > 0)
{
commandLength = response[USN3170Constants.DATA_LENGTH_INDEX] + 3;
}
}
streamBuffer = response.ToArray();
}
(int fromReader = ReaderStream.ReadByte();
被try/catch包围了,只捕获到异常中止的线程异常,所以我把它拿出来了)
上面的代码是我怀疑的问题所在。不过,奇怪的是,我有一个单元测试,我觉得它很好地模仿了 Microsoft 测试应用程序。
(仅供引用,QUADPORT 是 FTDI 芯片组)
PosExplorer posExplorer;
DeviceCollection smartCardRWs;
[Test]
public void TestQuadPortOpen()
{
posExplorer = new PosExplorer();
smartCardRWs = posExplorer.GetDevices(DeviceType.SmartCardRW, DeviceCompatibilities.CompatibilityLevel1);
//if using quadport one item is the MSR and the other is the RFID
//because of that one of them will fail. Currently the first Device in the collection is the the RFID, and the second is MSR
Assert.GreaterOrEqual(smartCardRWs.Count, 2);
//Hardware Id: QUADPORT\QUAD_SERIAL_INTERFACE
foreach(DeviceInfo item in smartCardRWs)
{
Assert.AreEqual("QUADPORT\\QUAD_SERIAL_INTERFACE", item.HardwareId);
}
SmartCardRW rfidDevice = (SmartCardRW)posExplorer.CreateInstance(smartCardRWs[0]);
SmartCardRW msrDevice = (SmartCardRW)posExplorer.CreateInstance(smartCardRWs[1]);
rfidDevice.Open();
Assert.AreNotEqual(ControlState.Closed, rfidDevice.State);
rfidDevice.Close();
try
{
msrDevice.Open();
Assert.Fail("MSR Device is not a RFID Device");
}
catch
{
Assert.AreEqual(ControlState.Closed, msrDevice.State);
}
rfidDevice = null;
msrDevice = null;
}
当我运行该测试时,我没有收到 SafeFileHandle 异常。事实上测试通过了。
所以我不知道如何追踪这个错误。由于我将在我创建的另一个程序中使用此服务对象,因此我可能最终会在该程序中使用此测试中的代码。但是我觉得 Microsoft Test App 或多或少是“黄金标准”。真的吗……应该不是。但它确实适合我的目的,所以我觉得这是我的代码的问题,而不是他们的。
关于如何缩小范围的任何技巧?仅供引用,我已经尝试使用调试器,但在运行开放代码时不会发生错误。我也走了更新状态计时器,它也没有抛出错误。一旦我点击继续,我就会得到异常。我打开了 Just My Code 和 Loaded Symbols,它告诉我“此模块的调试信息中缺少源信息”
最佳答案
这个问题(尤其是对 SerialPort 实例的引用)听起来很像 http://connect.microsoft.com/VisualStudio/feedback/details/140018/serialport-crashes-after-disconnect-of-usb-com-port 中记录的问题.
据我了解,在非永久性 SerialPort 的情况下(例如与 USB 设备相关联的端口),当端口意外“消失”时,与其相关联的底层 Stream 将被处置。如果在随后调用 SerialPort.Close 时端口上有事件的读或写操作可能会导致您提到的异常,但是该异常发生在运行在不同线程上的 Microsoft 代码中并且无法从您的内部捕获代码。 (它仍然会被您绑定(bind)到 AppDomain 上的 UnhandledException 事件的任何“最后机会”异常处理程序看到。)
链接文档中似乎有两种基本的变通方法。在这两种情况下,打开端口后,您都会为打开的端口存储对 BaseStream 实例的引用。然后,一种解决方法会禁止对该基本流进行垃圾收集。另一个在基本流上显式调用 Close,在调用 SerialPort 上的 Close 之前捕获该操作期间抛出的任何异常。
编辑:就其值(value)而言,在 .NET Framework V4.5 下,Microsoft Connect 站点上记录的解决方法似乎没有一个可以完全解决问题,尽管它们可能会降低频率它发生的地方。 :-(
关于c# - ObjectDisposedException : Safe handle has been closed,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20708641/
此错误显然源于 xlsxwriter。我不确定它来自我的代码的哪一行,因为每次我尝试调试时,我的编辑器 Visual Studio 2019 都会崩溃。但是,在使用 VPN 和远程桌面连接时,我在笔记
我有一个用于原型(prototype)的游戏数据表。我在工作时生成数据,但当我离开并且我的机器进入休眠状态时,数据生成停止。这导致我的元素收藏出现很大差距。 我希望能够移动表格的 DateTimeCr
我正在使用wavesurfer在我的网页上显示歌曲波形。我正在使用以下代码 - function setupSongwaves(songJson) { var songwaveid = '#s
我是 JDBC 新手... Student类有Constructor、add()、update()和delete()等方法... 在构造函数中打开连接。下面代码中的 conn.close() 和 ps
考虑以下代码,它是许多 ChannelFactory 示例的典型代码: WSHttpBinding myBinding = new WSHttpBinding(); EndpointAddress m
我正在阅读 Java Data Access — JDBC、JNDI 和 JAXP,了解 Connection、PooledConnection 接口(interface)。据我了解, PooledC
我正在做我的第一个 android 学习教程,但遇到了标题描述的这个错误..这是我试图在 Eclipse 上做的应用程序,java,这是我的代码..(代码是自动生成的由项目) package com.
我正在使用 JPA、Hibernate、Jboss 和容器管理事务。当我尝试用数据保存我的大实体时,它会抛出以下异常。将我的实体视为图形模型。这个异常并不是每次都会抛出。 ERROR [org.jbo
我有 GWT 应用程序,它与 AdaptivePayment API 上的灯箱集成。 我无法使用提供的代码关闭取消/返回页面: dgFlow = top.dgFlow || top.opener.to
即使我已经实现了上述方法 close(),Eclipse 仍向我显示上述错误。 代码如下: public void update_project(View view) { EditText c
在我的网络应用程序中,我广泛使用了数据库。 我有一个抽象的 servlet,所有需要数据库连接的 servlet 都继承自它。该抽象 servlet 创建一个数据库连接,调用必须由继承 servlet
我在这里看到很多答案都说要使用 close() 来销毁套接字,但我使用的指南来自 msdn让我使用 closesocket()。我想知道是否存在差异,是否有理由使用其中一种。 在这两种情况下,我都看到
我在 python 中使用 with 语句( PEP 343 ) 时遇到了一些问题,以便在上下文之后自动管理资源清理。特别是,with 语句 始终假定资源清理方法是 .close()。 IE。在下面的
在本地连接上调用 RTCPeerConnection.close() 时,我希望远程连接接收到 closed connectionstatechange 事件。 相反,几秒钟后出现disconnect
我正在使用 netty 3.6.6。 有人可以解释以下两个代码之间的区别吗? channel.close(); channel.write(ChannelBuffers.EMPTY_BUFFER).a
WebSocket.readyState可以是CONNECTING、OPEN、CLOSING或CLOSED。 CLOSING 和 CLOSED 状态有什么区别?为什么区分这两种状态很有用?我可以将 C
想象一下,您在 Python 中打开了某个文件(无论是用于读取、写入还是其他)。我刚刚注意到,当您想关闭该文件时,您可以输入: somefile.close() 或者您可以输入: somefile.c
我在我的应用程序的各种类和线程中打开、访问、写入等数据库。我有一个数据库 self.run_params["db"] 我在整个应用程序中都使用它来访问。 问题 1:我是否应该在每次访问后关闭光标? 问
我正在尝试创建一个 vanilla JavaScript 模态,当从 HTML 文件(或 JS 文件)实例化它时,它具有由用户自定义的能力。但是,在处理关闭模式的 close() 函数时,不是一次关闭
所以这可能是一个菜鸟类型的问题,但这就是我想知道的。 假设我有两个屏幕,第一个屏幕是 idk,例如 Screen1。假设用户在 Screen1 上点击了OK,这会将他们带到Screen2。 我目前正在
我是一名优秀的程序员,十分优秀!