问候开发者!
由于 SO 几乎总是对我的编程问题有所帮助,因此我决定注册并尝试解决我最近遇到的问题。这真是一个我和我的同事都无法弄清楚的奇怪现象。很抱歉,我无法提供工作示例,但该项目的分解方式很复杂,并且需要特定的硬件才能正常运行。所以我会尽力解释。
我们项目的基础是一个本地库(在本例中为 32 位 Windows C-DLL),用于通过 Java 应用程序 ( JNA ) 访问项目特定的硬件。目的是在 Swing UI 中管理和显示硬件的专有文件系统(通过 USB 连接)。这对我们来说是一个非常常见的项目配置,因为我们在 Java 应用程序中集成了很多本机库和驱动程序。
摘要:用于枚举设备的单元测试工作正常。本机库的一个模块分配内存并用结构填充它,每个结构都包含连接设备的信息。这不是好的做法,但由于我们对这部分没有任何影响,所以我们必须接受它。我在 Java/JNA 中映射了这个结构,调用原生函数,将结构内容复制到 Java 传输类并在控制台中打印。工作得很好。
现在,如果在枚举设备时有激活的 UI 操作,本机库会因访问冲突而崩溃。即使这个 UI 操作与库无关。JNA 错误消息显示一个 EXCEPTION_ACCESS_VIOLATION (0xc0000005),SO 研究显示为无效/空内存。
有没有人遇到过这样的问题?我们当然从来没有这样做过。我花了几天时间将错误源缩小到这部分代码。涉及本机库时,调试并不容易。是否可能存在JVM内存并发问题?由于本机库自己分配内存而 JVM 对此一无所知 - 所以 JVM 会尝试在已使用的内存中为新的 Swing 组件分配内存?
代码: 以下代码片段来自我的单元测试,尽可能分解。预期的顺序很明显:从根节点中删除节点,加载连接的设备并将这些设备添加为新节点。此代码因访问冲突而崩溃,但不是在本机调用时崩溃 - 只要我访问树组件,它就会崩溃。
public void loadDevices(){
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
rootNode.removeAllChildren();
rootNode.add(new LoadingNode());
tree.expandPath(new TreePath(rootNode));
}
});
final List<Device> devices = lib.loadDevices(); // wrapped native call
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
rootNode.removeAllChildren();
if(!devices.isEmpty()){
for (Device dev : devices ) {
DevNode node = new DevNode(dev);
rootNode.add(node);
}
}
}
});
}
注意:DevNode不包含任何原生数据,每个原生结构的内容都被复制到一个 Java 传输对象中。GC 在尝试移动对象数据时应该不会出现问题,因为所有非托管代码都在lib#loadDevices()方法中本地处理。
当我完全删除对SwingUtilities的调用并将生成的设备信息打印到控制台而不是创建节点时,这部分工作正常。一旦我尝试访问JTree或TreeModel成员,代码就会崩溃。如果我在调用SwingUtitilies#invokeLater()或在同一个线程中执行此操作,它不会发生。
我知道这是一个非常具体的问题,几乎没有人会感兴趣(这使得在 SO/Google 中搜索解决方案变得非常困难)。但也许我很幸运,有人已经遇到过这个问题。
这么长的
xander
编辑:最初这段代码被包装在一个工作线程中,导致相同的结果。这只是我的单元测试的一个片段。
编辑2:对不起,我似乎没有让自己足够清楚或忘记在这里提到一些重要的事情。对树或其模型的访问不一定与本机库有关。再看一下代码:第一次调用 invokeLater 只是从树中删除节点。即使我删除了对 invokeLater 的第二次调用,本机库也会崩溃!