6
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;

class Task implements Callable<String> {
    public String call() throws Exception {
        String s = "initial";
        try {
            System.out.println("Started..");
            /*for (int i=0;i<10000;i++) {
                if (i % 2 == 0) {
                    System.out.println("Even");
                }
            }*/
            boolean flag = true;
            while(flag) {

            }
            System.out.println("Finished!");
            s = "Done";
        }
        catch (RuntimeException e) {
            s = "RuntimeException";
        }
        catch (Exception e) {
            s = "Exception";
        }
        finally {

        }
        return s;
    }
}

public class ExecutorServiceTest {

    public static void main(String[] args) throws Exception {
        ExecutorService executor = Executors.newSingleThreadExecutor();
        List<Future<String>> result = executor.invokeAll(Arrays.asList(new Task()), 5, TimeUnit.SECONDS);
        executor.shutdown();
        Iterator<Future<String>> iter = result.iterator();
        while (iter.hasNext()) {
            System.out.println("Came here");
            Future<String> fut = iter.next();
            System.out.println(fut.get());
        }
    }
}

スレッドが無限ループを実行するのを停止する方法はありますか?

4

2 に答える 2

7

flagはい、 (または論理的&&に)をに置き換えることができます!Thread.currentThread().isInterrupted()

このように、タスクがキャンセルされると、ループは終了します。

ループは次のようになります。

while(!Thread.currentThread().isInterrupted() && flag) {
  /* Do work. */
}

使用は次のようになります。

ExecutorService executor = Executors.newSingleThreadExecutor();
Future<String> task = executor.submit(new Task());
String str;
try {
  str = task.get(5, TimeUnit.SECONDS);
} finally {
  task.cancel(true);
}
于 2010-03-01T18:44:59.240 に答える
0

内部synchronized (this) { this.wait() }の代わりにを使用することを検討してください。ブール値を外部に設定する場合(おそらく直接またはメソッドを介して。直接アクセスではフラグ変数がであることを確認してください) 、スリープしているスレッドをウェイクアップするために呼び出します(タスクオブジェクトがローカル変数であることを確認してください)。匿名にする代わりに、メソッドを呼び出して、フラグを)内のクラス属性にすることができます。sleepcall()flagflag()volatiletask.notifyAll()Task

また、ループが不必要にサイクルを浪費するため、このように効率的になります。正確なメカニズムは「保護されたブロック」と呼ばれます(http://java.sun.com/docs/books/tutorial/essential/concurrency/guardmeth.html)。待機から復帰したら、フラグ変数をテストして、設定されていることを確認します。

編集:元の質問をより詳しく調べ、既存のコードと原則を使用して例を作成しました(猫の皮を剥ぐ方法は複数あります:))。これを試してください-タイムアウトのためにキャンセルされた現在のスレッドの中断されたステータスのために、ここのループは終了します:

package ett;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;

class Task implements Callable<String> {
    public String call() throws Exception {
        String s = "initial";
        System.out.println("Started..");
        for (int i=0;;i++) {
            if (i % 2 == 0) {
                System.out.println("Even");
            }
            Thread.yield();
            if (Thread.interrupted()) break;
        }
        System.out.println("Finished!");
        s = "Done";

        return s;
    }
}

public class ExecutorServiceTest {

    public static void main(String[] args) throws Exception {
        ExecutorService executor = Executors.newSingleThreadExecutor();
        List<Future<String>> result = executor.invokeAll(Arrays.asList(new Task()), 1, TimeUnit.SECONDS);
        executor.shutdown();

        System.out.println("came here");

        for (Future<String> f : result) {
            try {
                System.out.println(f.get());
            } catch (CancellationException e) {
                e.printStackTrace();
            }
        }
    }
}
于 2010-03-01T18:53:40.023 に答える