////////////////////////////////////////////////////////////////////////////////
// << j841d.java >>
//
//  データ構造(4):集合(TreeSetクラス、同期処理 対策1)
//
//  TreeSetクラスはスレッドに対して安全でない。
//  複数のスレッドがTreeSetオブジェクトに同時にアクセスし、
//  要素の追加や削除を行なったとき、結果が保証されない。
//
//  対策法1:synchronizedブロックを使って、同期を保証する。
//
//    集合に対して1000個の要素の追加・削除をn回繰り返すスレッドm個を実行させて、
//  最後に集合が空になるかどうか調べる。
//  同期がとれていれば空集合が得られるが、失敗すれば、空でない集合が
//  得られるか、停止しなくなる。
//
////////////////////////////////////////////////////////////////////////////////
 
import java.util.*;

// スレッド(AR)の定義。
class ARThread extends Thread {
  TreeSet ts;
  int n;
  ARThread(TreeSet ts, int n) { // コンストラクタ。 
    this.ts = ts; this.n = n;
  }
  // 処理本体。
  public void run() {
    System.out.println(Thread.currentThread().getName() + ":処理開始 " + ts.size());
    try {
      for( int i=1; i<=n; i++ ) { 
        synchronized(ts) {
          for( int j=1; j<=1000; j++ ) { ts.add(new Integer(i)); }
          for( int j=1; j<=1000; j++ ) { ts.remove(new Integer(i)); }
        }
      }
    } catch( Exception e ) {
      System.out.println(e);
    }
    System.out.println(Thread.currentThread().getName() + ":処理終了 " + ts.size());
  }
}

class j841d {

  public static void main(String args[]) {

    int m = Integer.parseInt(args[0]); // args[0]:スレッドの個数。
    int n = Integer.parseInt(args[1]); // args[1]:繰り返し回数。
    System.out.println("m = " + m);
    System.out.println("n = " + n);

    // スレッドに対して安全でない空の集合を作成。
    TreeSet ts = new TreeSet();

    // スレッドの作成と実行。
    ARThread arth[] = new ARThread[m];
    for( int i=0; i<m; i++ ) {
      arth[i] = new ARThread(ts,n); arth[i].start();
    }

    // 集合処理の確認。
    try {
      for( int i=0; i<m; i++ ) { arth[i].join(); } 
      System.out.println("集合:" + ts.size());
    } catch ( Exception e ) {
      System.out.println(e); 
    }

  }

}
実行結果
% javac j841d.java
注:j841d.java の操作は、未チェックまたは安全ではありません。
注:詳細については、-Xlint:unchecked オプションを指定して再コンパイルしてください

% java j841d 5 1000
m = 5
n = 1000
Thread-0:処理開始 0
Thread-1:処理開始 0
Thread-2:処理開始 0
Thread-3:処理開始 0
Thread-4:処理開始 0
Thread-4:処理終了 0
Thread-2:処理終了 0
Thread-1:処理終了 0
Thread-3:処理終了 0
Thread-0:処理終了 0
集合:0

% java j841d 5 10000
m = 5
n = 10000
Thread-0:処理開始 0
Thread-1:処理開始 0
Thread-2:処理開始 0
Thread-3:処理開始 0
Thread-4:処理開始 0
Thread-1:処理終了 0
Thread-4:処理終了 0
Thread-3:処理終了 0
Thread-0:処理終了 0
Thread-2:処理終了 0
集合:0