关于运行GUI时线程的问题

时间:2019-02-12 22:27:38

标签: java swingworker

Im目前正在为Uni编写一个程序,该程序使用RFID跟踪器来跟踪RFID标签通过时的运动。这模仿了钱包在穿过房间时的移动。

我在GUI中添加了一个按钮,该按钮可初始化跟踪器并在标签通过跟踪器时更新我的​​数据库。他们保持在线60秒,一切正常。但是,一旦按下此按钮,我的GUI将冻结,直到该进程停止。这样可以防止GUI上的表更新。

我意识到我需要使用线程,我发现SwingWorker看起来不错,但是经过一段时间的尝试,我似乎无法弄清楚。

我想知道是否有比我有更多经验的人可以告诉我SwingWorker是否可以满足我的需要,并给我一些使用建议。

以下是我的代码片段:

跟踪JButton:

JButton trackingButton = new JButton("Start Tracking");
trackingButton.addActionListener(new ActionListener() {
    public void actionPerformed(ActionEvent e) {
        try {
            RFIDHandler.openRFID(); //runs the RFID readers
        } catch (PhidgetException | SQLException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        }
    }
});

我的openRFID方法的开始:

public static void openRFID() throws PhidgetException, SQLException {   

    RFID phid1 = new RFID();
    RFID phid2 = new RFID();
    WalletDAO dao = new WalletDAO();

    // Open and start detecting rfid cards
    phid1.open(5000);  // wait 5 seconds for device to respond
    phid2.open(5000);  // wait 5 seconds for device to respond


    phid1.setAntennaEnabled(true);
    phid2.setAntennaEnabled(true);

    // Make the RFID Phidget able to detect loss or gain of an rfid card
    phid1.addTagListener(new RFIDTagListener() {
        public void onTag(RFIDTagEvent e) {
            try {
                Wallet w = dao.getWallet(e.getTag());                   
                String location = w.getLocation();
                System.out.println(location);

                if(Objects.equals(location, "Room A")) {
                    System.out.println(w.getName() + "'s Wallet read");
                    w.setLocation("Room B");
                    try {
                        dao.updateLocation(w);
                        WalletLocatorInterface.refreshTable(); // refreshes table on GUI
                        System.out.println("Currently in room B");
                    } catch (SQLException e1) {
                        // TODO Auto-generated catch block
                        e1.printStackTrace();
                    }
                }
                // ... else statement + another TagListener 

    System.out.println("Gathering data for 60 seconds\n\n");
    try {
        Thread.sleep(60000);
    } catch (InterruptedException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    }

    phid1.close();
    phid2.close();
    System.out.println("\nClosed RFIDs");
}

谢谢,让我知道您是否需要我清理所有东西。

2 个答案:

答案 0 :(得分:1)

您的问题是func。您可以使用ScheduledExecutorService完成此处的操作。只需将您的代码修改为此:

Thread.sleep

(顺便说一下,我使用了一个lambda表达式,可以here进行解释。)

关于... System.out.println("Gathering data for 60 seconds\n\n"); try { Thread.sleep(60000); } catch (InterruptedException e1) { // TODO Auto-generated catch block e1.printStackTrace(); } Executors.newScheduledThreadPool(1).schedule(() -> { phid1.close(); phid2.close(); System.out.println("\nClosed RFIDs"); }, 60, TimeUnit.SECONDS); } ,我经验不足。但是,使用SwingWorker,您可以在后台执行操作,并且在更新时不会使GUI变得混乱。 但是,如果您没有像@MadProgrammer所说的那样真正更新GUI,那么使用SwingWorker并不会给您带来真正的好处。

答案 1 :(得分:0)

当您尝试设计类似的东西时,您需要以一种更加独立的方式重新思考您的方法。

我可能要做的第一件事是定义一个可以传递信息的委托,这使它更易于处理,因为您无需为SwingWorker s PropertyChangeListener和它允许您解耦代码...

public interface RFIDWorkerDelegate {
    public void locationsUpdated(List<String> locations);
}

接下来,您需要执行可能会阻塞UI的操作,并将其放入SwingWorker s doInBackfground中。

然后,您需要确定最佳时间以所需的更改来更新UI。然后,您调用publish以“发布”结果。这些将传递给process方法,该方法在事件调度线程的上下文中调用,例如...

public class RFIDWorker extends SwingWorker<Void, String> {

    private RFIDWorkerDelegate delegate;

    public RFIDWorker(RFIDWorkerDelegate delegate) {
        this.delegate = delegate;
    }

    @Override
    protected void process(List<String> chunks) {
        delegate.locationsUpdated(chunks);
    }

    @Override
    protected Void doInBackground() throws Exception {
        RFID phid1 = new RFID();
        RFID phid2 = new RFID();
        WalletDAO dao = new WalletDAO();

        // Open and start detecting rfid cards
        phid1.open(5000);  // wait 5 seconds for device to respond
        phid2.open(5000);  // wait 5 seconds for device to respond

        phid1.setAntennaEnabled(true);
        phid2.setAntennaEnabled(true);

        // Make the RFID Phidget able to detect loss or gain of an rfid card
        phid1.addTagListener(new RFIDTagListener() {
            public void onTag(RFIDTagEvent e) {
                Wallet w = dao.getWallet(e.getTag());
                String location = w.getLocation();
                System.out.println(location);

                publish(location);

                if (Objects.equals(location, "Room A")) {
                    System.out.println(w.getName() + "'s Wallet read");
                    w.setLocation("Room B");
                    try {
                        dao.updateLocation(w);
                        WalletLocatorInterface.refreshTable(); // refreshes table on GUI
                        System.out.println("Currently in room B");
                    } catch (SQLException e1) {
                        // TODO Auto-generated catch block
                        e1.printStackTrace();
                    }
                }
            }
        });

        System.out.println("Gathering data for 60 seconds\n\n");
        try {
            Thread.sleep(60000);
        } catch (InterruptedException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        }

        phid1.close();
        phid2.close();
        System.out.println("\nClosed RFIDs");
        return null;
    }
}

那么您可能会称其为类似...

new RFIDWorker(new RFIDWorkerDelegate() {
    @Override
    public void locationsUpdated(List<String> locations) {
        // Update the UI as required
    }
});