0

我是线程新手,正在尝试复制一个在我的网络应用程序中中断线程的简单示例。我有下面的类(ComputeResults 有其他变量和函数、setter/getter 等,但这是我无法工作的新代码):

@ManagedBean(name="results")
@RequestScoped
public class ComputeResults implements Serializable{

    Thread scan;
    public void testrun() {
    scan = new Thread(new Runnable() {
        @Override
        public void run() {
            int i = 0;
            while (!Thread.currentThread().isInterrupted()) {
                try {
                    i++;
                    if (i == 1) {
                        scan.interrupt();
                    }
                } 
                catch (Exception e) {
                    Thread.currentThread().interrupt();
                }
                catch (Throwable t) {
                    System.out.println("Thrown test: "+t.getMessage());
                }
            }
        }
    });

    scan.start();
}

    public void stoprun() {
        if(scan != null){
            scan.interrupt();
        }
    }
}

在我的界面中,我有一个按钮来启动线程:

<p:commandLink action="submit" value="" onclick="testdialog.show()" oncomplete="testdialog.hide()" actionListener="#{results.testrun}" update="messages, runmsg, @form results" />

还有一个试图打断它:

<p:commandButton action="submit" value="Cancel Test" onclick="testdialog.hide()" actionListener="#{results.stoprun}" update="messages, runmsg" />

问题是“停止运行”功能将“扫描”视为空,我不知道为什么。在 testrun() 中添加 scan.interrupt() 可以正常工作。我考虑过使用 Thread.currentThread().interrupt() 但是当我调用 stoprun 时,当前线程 ID/名称似乎不同。

4

2 回答 2

2

那是因为 bean 是@RequestScoped- 每个 HTTP 请求(= 按钮单击)都会获得一个新实例。

@Scope("session")至少需要做到。

于 2012-08-22T07:34:52.597 回答
0

stopRun在第一次运行之前被调用到testRun,因此该成员是null。这是因为每个方法调用都发生在一个新实例上。

于 2012-08-22T07:36:30.187 回答