首页>Program>source

我正在用Java Swing编写应用程序.我需要的是一个可以使用图形界面中的按钮停止"阐述"线程的过程。

这是一个专注于我需要的简单项目

import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JTextArea;
/*
 * To change this template, choose Tools | Templates
 * and open the template in the editor.
 */
/**
 *
 * @author Nikola
 */
public class Main extends javax.swing.JFrame
{
    private MyThread THREAD;
    public Main()
    {
        initComponents();
    }
    @SuppressWarnings("unchecked")
    // <editor-fold defaultstate="collapsed" desc="Generated Code">
    private void initComponents() {
        jButton1 = new javax.swing.JButton();
        jScrollPane1 = new javax.swing.JScrollPane();
        jTextArea1 = new javax.swing.JTextArea();
        jButton2 = new javax.swing.JButton();
        jButton3 = new javax.swing.JButton();
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        jButton1.setText("Pause Thread");
        jButton1.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jButton1ActionPerformed(evt);
            }
        });
        jTextArea1.setColumns(20);
        jTextArea1.setRows(5);
        jScrollPane1.setViewportView(jTextArea1);
        jButton2.setText("Resume Thread");
        jButton2.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jButton2ActionPerformed(evt);
            }
        });
        jButton3.setText("Start Thread");
        jButton3.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jButton3ActionPerformed(evt);
            }
        });
        javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane());
        getContentPane().setLayout(layout);
        layout.setHorizontalGroup(
            layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGroup(layout.createSequentialGroup()
                .addContainerGap()
                .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                    .addGroup(layout.createSequentialGroup()
                        .addComponent(jButton3)
                        .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED, 63, Short.MAX_VALUE)
                        .addComponent(jButton2)
                        .addGap(18, 18, 18)
                        .addComponent(jButton1))
                    .addComponent(jScrollPane1))
                .addContainerGap())
        );
        layout.setVerticalGroup(
            layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGroup(javax.swing.GroupLayout.Alignment.TRAILING, layout.createSequentialGroup()
                .addContainerGap()
                .addComponent(jScrollPane1, javax.swing.GroupLayout.DEFAULT_SIZE, 244, Short.MAX_VALUE)
                .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.UNRELATED)
                .addGroup(layout.createParallelGroup(javax.swing.GroupLayout.Alignment.BASELINE)
                    .addComponent(jButton1)
                    .addComponent(jButton2)
                    .addComponent(jButton3))
                .addContainerGap())
        );
        pack();
    }// </editor-fold>
    private void jButton3ActionPerformed(java.awt.event.ActionEvent evt)
    {
        THREAD = new MyThread(jTextArea1);
        THREAD.start();
    }
    private void jButton1ActionPerformed(java.awt.event.ActionEvent evt)
    {
        try
        {
            THREAD.pauseThread();
        }
        catch (InterruptedException ex)
        {
            ex.printStackTrace();
        }
    }
    private void jButton2ActionPerformed(java.awt.event.ActionEvent evt)
    {
        THREAD.resumeThread();
    }
    public static void main(String args[])
    {
        /*
         * Set the Nimbus look and feel
         */
        //<editor-fold defaultstate="collapsed" desc=" Look and feel setting code (optional) ">
        /*
         * If Nimbus (introduced in Java SE 6) is not available, stay with the
         * default look and feel. For details see
         * http://download.oracle.com/javase/tutorial/uiswing/lookandfeel/plaf.html
         */
        try
        {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels())
            {
                if ("Nimbus".equals(info.getName()))
                {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        }
        catch (ClassNotFoundException ex)
        {
            java.util.logging.Logger.getLogger(Main.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        catch (InstantiationException ex)
        {
            java.util.logging.Logger.getLogger(Main.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        catch (IllegalAccessException ex)
        {
            java.util.logging.Logger.getLogger(Main.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        catch (javax.swing.UnsupportedLookAndFeelException ex)
        {
            java.util.logging.Logger.getLogger(Main.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        //</editor-fold>
        /*
         * Create and display the form
         */
        java.awt.EventQueue.invokeLater(new Runnable()
        {
            public void run()
            {
                new Main().setVisible(true);
            }
        });
    }
    // Variables declaration - do not modify
    private javax.swing.JButton jButton1;
    private javax.swing.JButton jButton2;
    private javax.swing.JButton jButton3;
    private javax.swing.JScrollPane jScrollPane1;
    private javax.swing.JTextArea jTextArea1;
    // End of variables declaration
}
class MyThread extends Thread
{
    JTextArea area;
    private final Object lock = new Object();
    public MyThread(JTextArea area)
    {
        super();
        this.area = area;
    }
    @Override
    public void run()
    {
        for(int i=0 ; ; i++)
            area.setText(i+"");
    }
    public void pauseThread() throws InterruptedException
    {
        synchronized(lock)
        {
            lock.wait();
        }
    }
    public void resumeThread()
    {
        synchronized(lock)
        {
            lock.notify();
        }
    }
}

问题很简单: 在实际的应用程序中,用户设置一些选项,然后启动线程,对选定的数据进行详细说明。

我想提供一个"暂停"按钮,以便用户可以暂时停止详细说明并进行必要的检查,然后可以恢复操作。

我编码的方式是停止的图形线程,而不是"精化"线程。

如果运行示例代码并按"开始",则文本区域开始计数.我需要的最终结果是,当我按下"暂停"按钮时,线程进入"睡眠"状态,计数停止;当我按下"恢复"按钮时,线程"唤醒",文本区域中的计数开始

最新回答
  • 5月前
    1 #

    您无法明确地以您想要的方式暂停另一个线程。

    您需要做的是signal,另一个线程应该通过设置某种标志来停止.有问题的线程必须具有逻辑来检查该标志并在发生这种情况时暂停其工作。

    因此在这种情况下,也许可以更改 MyThread 如下:

    class MyThread extends Thread {
        private volatile boolean running = true; // Run unless told to pause
        ...
        @Override
        public void run()
        {
            for(int i=0 ; ; i++)
            {
                // Only keep painting while "running" is true
                // This is a crude implementation of pausing the thread
                while (!running)
                    yield;
                area.setText(i+"");
        }
        public void pauseThread() throws InterruptedException
        {
            running = false;
        }
        public void resumeThread()
        {
            running = true;
        }
    }
    

    这是一个粗略的例子,为简洁起见,它使用了一种自旋锁而不是基于监视器的适当睡眠.希望它传达了您如何使用标志来控制线程暂停的想法。

    请注意,如果您在块中执行了一些长时间运行的步骤,而不仅仅是 setText 电话,检查 Thread.currentThread().interrupted()是一个好习惯 在每个步骤之间-如果设置了itnerrupt标志,则退出循环.这大致就是内置的阻塞方法(例如I / O)所做的,以便它们可以被其他线程中断-因为 running 标记仅在每个循环中检查一次,如果每个循环需要20分钟,则对它进行设置没有多大好处。

  • 5月前
    2 #

    像这样尝试:

    class MyThread extends Thread {
        JTextArea area;
        private final Object GUI_INITIALIZATION_MONITOR = new Object();
        private boolean pauseThreadFlag = false;
        public MyThread(JTextArea area) {
            super();
            this.area = area;
        }
        @Override
        public void run() {
            for(int i=0 ; ; i++) {
                checkForPaused();
                area.setText(i+"");
            }
        }
        private void checkForPaused() {
            synchronized (GUI_INITIALIZATION_MONITOR) {
                while (pauseThreadFlag) {
                    try {
                        GUI_INITIALIZATION_MONITOR.wait();
                    } catch (Exception e) {}
                }
            }
        }
        public void pauseThread() throws InterruptedException {
            pauseThreadFlag = true;
        }
        public void resumeThread() {
            synchronized(GUI_INITIALIZATION_MONITOR) {
                pauseThreadFlag = false;
                GUI_INITIALIZATION_MONITOR.notify();
            }
        }
    }
    

    最好像您一样使用监视器.但是您不能从外面强制等待.您必须告诉线程等待,直到再次(通过监视器)通知他为止.在这种情况下,您只有这种简单的方法 checkForPaused() 您将不得不摆上重要的位置,这样就不会有很长的延迟,直到线程暂停为止。

    您还可以扩展此功能,以便可以询问线程是否因在 checkForPaused()中设置的标志而被暂停 和一个 public boolean isPausing()

  • cuda:如何实现涉及多个变量的自定义原子函数?
  • c:windows命令提示符中的EOF不会终止输入流