为此,您可以使用java.util.concurrent 中的功能
在包含 UI 的现有类中,您必须添加类似于以下内容的内容:
//class variable to store the future of your task
private Future<?> taskFuture = null;
//to be called from button "Start" action handler
public void actionStart() {
//don't double start, if there is one already running
if(taskFuture == null || taskFuture.isDone()) {
//create the new runnable instance, with the proper commands to execute
MyShellExecutor ex = new MyShellExecutor(new String[] { "sh",testPath + "/install.sh", cmd, "&" });
//we only need one additional Thread now, but this part can be tailored to fit different needs
ExecutorService newThreadExecutor = Executors.newSingleThreadExecutor();
//start the execution of the task, which will start execution of the shell command
taskFuture = newThreadExecutor.submit(ex);
}
}
//to be called from button "Stop" action handler
public void actionStop() {
//if not already done, or cancelled, cancel it
if(taskFuture !=null && !taskFuture.isDone()) {
taskFuture.cancel(true);
}
}
完成这项工作的主要组件是Runnable
我命名的组件MyShellExecutor
,如下所示:
public class MyShellExecutor implements Runnable {
//stores the command to be executed
private final String[] toExecute;
public MyShellExecutor(String[] toExecute) {
this.toExecute=toExecute;
}
public void run() {
Runtime runtime = Runtime.getRuntime();
Process process = null;
try {
process = runtime.exec(toExecute);
int exitValue = process.waitFor();
System.out.println("exit value: " + exitValue);
BufferedReader buf = new BufferedReader(new InputStreamReader(process.getInputStream()));
String line = "";
while ((line = buf.readLine()) != null) {
System.out.println("exec response: " + line);
//do whatever you need to do
}
} catch (InterruptedException e) {
//thread was interrupted.
if(process!=null) { process.destroy(); }
//reset interrupted flag
Thread.currentThread().interrupt();
} catch (Exception e) {
//an other error occurred
if(process!=null) { process.destroy(); }
}
}
}
注意:运行耗时操作时,一定不要在UI线程上做。这只会阻止用户,并不能提供良好的用户体验。总是做一些可能使用户在不同线程上等待的事情。
推荐阅读:Java 并发实践