2

问候开发者!

由于 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的调用并将生成的设备信息打印到控制台而不是创建节点时,这部分工作正常。一旦我尝试访问JTreeTreeModel成员,代码就会崩溃。如果我在调用SwingUtitilies#invokeLater()或在同一个线程中执行此操作,它不会发生。

我知道这是一个非常具体的问题,几乎没有人会感兴趣(这使得在 SO/Google 中搜索解决方案变得非常困难)。但也许我很幸运,有人已经遇到过这个问题。

这么长的
xander

编辑:最初这段代码被包装在一个工作线程中,导致相同的结果。这只是我的单元测试的一个片段。

编辑2:对不起,我似乎没有让自己足够清楚或忘记在这里提到一些重要的事情。对树或其模型的访问不一定与本机库有关。再看一下代码:第一次调用 invokeLater 只是从树中删除节点。即使我删除了对 invokeLater 的第二次调用,本机库也会崩溃!

4

2 回答 2

1
final List<Device> devices = lib.loadDevices();
  • (假设您的代码使用 JNA 能够单独返回每个节点或List<Device>'s在结束时返回一次集合)应该从工作线程调用,例如 Runnable#Thread 或 SwingWorker

  • 工作线程的所有输出都直接添加到 DefaultTreeModel(或通过创建新的 MutableTreeNode)应包装到 invokeLater

  • 如果不发布SSCCEList<Device>'s可能是 USB 端口列表),不知道会发生什么,简短、可运行、可编译,仅关于 JTree,是模型和 JNI/JNA ...

于 2013-04-02T15:35:37.663 回答
1

在我了解到这一点之前,我在 JTree 上遇到了很多困难:

在 JTree 上,不应编辑节点本身,而应使用 DefaultTreeModel 上提供的方法:

setRoot(TreeNode root) 
removeNodeFromParent(MutableTreeNode node) 
insertNodeInto(MutableTreeNode newChild, MutableTreeNode parent, int index) 

编辑节点本身可能(迟早会)导致奇怪的行为。

这当然是在您使用 DefaultTreeModel 和 MutableTreeNode 时。我强烈建议这样做,因为我已经看到了 TreeModel 的许多错误实现。

于 2013-04-02T20:28:37.240 回答