java - 从应用程序上下文返回 ObservableList<String>

标签 java multithreading sqlite javafx-2

我目前正在使用嵌入在应用程序类中的静态数据库类,它工作得很好。但是,当我返回ObservableList<String>时从函数返回并尝试将其设置为它抛出的 ListView Caused by: java.lang.reflect.InvocationTargetException 。我知道这是一个线程问题,在这种情况下我只是无法弄清楚如何将值从应用程序线程传递回 UI 线程。我是 JavaFX 新手。

Controller 类:

public class SampleController implements Initializable {

    @FXML
    public ListView<String> lvGroup;

    @FXML
    private void OpenDbClick(ActionEvent event){
        System.out.println("open db");
        ObservableList<String> ol;
        ol = WeddingGuestList.DatabaseConn.getGroups();
        lvGroup.setItems(ol);   <-error occurs here
    }
}

数据库方法:

public static ObservableList<String> getGroups(){
    ObservableList<String> groups = FXCollections.observableArrayList();

    try{
       Statement stmt = con.createStatement();
       stmt.setQueryTimeout(30);
       ResultSet rs = stmt.executeQuery("Select * from GuestGroup");
       while (rs.next()){
           groups.add(rs.getString("Name"));
       }

    }catch (SQLException e){
        e.printStackTrace();
    }catch (Exception e){
        e.printStackTrace();
    }
    return groups;
}

我不明白如何将数据返回到 ListView 中。

编辑:

堆栈跟踪:

Exception in thread "JavaFX Application Thread" java.lang.RuntimeException: java.lang.reflect.InvocationTargetException
    at javafx.fxml.FXMLLoader$ControllerMethodEventHandler.handle(FXMLLoader.java:1440)
    at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:69)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:217)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:170)
    at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:38)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:37)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:53)
    at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:28)
    at javafx.event.Event.fireEvent(Event.java:171)
    at javafx.scene.Node.fireEvent(Node.java:6863)
    at javafx.scene.control.Button.fire(Button.java:179)
    at com.sun.javafx.scene.control.behavior.ButtonBehavior.mouseReleased(ButtonBehavior.java:193)
    at com.sun.javafx.scene.control.skin.SkinBase$4.handle(SkinBase.java:336)
    at com.sun.javafx.scene.control.skin.SkinBase$4.handle(SkinBase.java:329)
    at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:64)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:217)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:170)
    at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:38)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:37)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:35)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:92)
    at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:53)
    at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:33)
    at javafx.event.Event.fireEvent(Event.java:171)
    at javafx.scene.Scene$MouseHandler.process(Scene.java:3328)
    at javafx.scene.Scene$MouseHandler.process(Scene.java:3168)
    at javafx.scene.Scene$MouseHandler.access$1900(Scene.java:3123)
    at javafx.scene.Scene.impl_processMouseEvent(Scene.java:1563)
    at javafx.scene.Scene$ScenePeerListener.mouseEvent(Scene.java:2265)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:250)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:173)
    at java.security.AccessController.doPrivileged(Native Method)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler.handleMouseEvent(GlassViewEventHandler.java:292)
    at com.sun.glass.ui.View.handleMouseEvent(View.java:528)
    at com.sun.glass.ui.View.notifyMouse(View.java:922)
    at com.sun.glass.ui.gtk.GtkApplication._runLoop(Native Method)
    at com.sun.glass.ui.gtk.GtkApplication$3$1.run(GtkApplication.java:82)
    at java.lang.Thread.run(Thread.java:724)
Caused by: java.lang.reflect.InvocationTargetException
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at sun.reflect.misc.Trampoline.invoke(MethodUtil.java:75)
    at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at sun.reflect.misc.MethodUtil.invoke(MethodUtil.java:279)
    at javafx.fxml.FXMLLoader$ControllerMethodEventHandler.handle(FXMLLoader.java:1435)
    ... 51 more
Caused by: java.lang.NullPointerException
    at testguesslist.SampleController.OpenDbClick(SampleController.java:56)
    ... 61 more

最佳答案

我不认为这就是导致上述异常的原因,但是:

理想情况下,您应该在单独的线程中进行数据库调用,因为这可能需要时间,并且您不想阻塞 JavaFX 线程。
执行此操作的方法是将 OpenDbClick 方法中正在完成的代码放入 javafx.concurrent.Task 中(检查 http://docs.oracle.com/javafx/2/threads/jfxpub-threads.htm )。

`
公共(public)类 SampleController {

    public ListView<String> lvGroup;
    private ExecutorService threadPool;



    private void OpenDbClick(ActionEvent event) {
        Task<Void> task = new Task<Void>() {

            @Override
            protected Void call() throws Exception {
                System.out.println("open db");
                final ObservableList<String> ol = WeddingGuestList.DatabaseConn.getGroups();
                Platform.runLater(new Runnable() {

                    @Override
                    public void run() {
                        lvGroup.setItems(ol);
                    }
                }); 
                return null;
            }
        };
        getPool().execute(task); 
    }

    private ExecutorService getPool() {
        if (threadPool == null) {
            threadPool = Executors.newCachedThreadPool();
        }
        return threadPool;
    }
}

`

Platorm.runLater 将强制在 JavaFX 线程而不是任务线程上执行其代码。 这很重要,因为设置 listview 的项目列表将导致场景图的更新,这应该在 UI 线程上完成。如果不使用Platform.runLater代码,上述将导致异常。

关于java - 从应用程序上下文返回 ObservableList<String>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18692921/

相关文章:

java - 为什么我不能在 Java 中使用两个线程持有同一个套接字?

java - 删除所有文件和子目录,但在 Java 中将当前目录保持为空

java - 从 angularJs 调用 java Controller

multithreading - 在 Azure 中执行多线程的正确方法是什么

java - Android 中的套接字连接错误

database - flutter 中使用 sqflite 保存嵌套模型的有效方法

android - 将大价格值与小数点一起存储在SQLite数据库中

java - 如何在 Linux 上使用 plesk 为基于服务的 tomcat 版本设置 JAVA_OPTS?

java - 在java中将map的map转换为map的列表

sqlite - 使用Sqlite3创建完整标题的标准版本