使用JAVA中的信号量控制线程访问

时间:2012-12-17 23:25:39

标签: java multithreading semaphore

我正在尝试运行此java代码,但它无法正常工作。

请让我知道我做错了什么。

for循环我小于10;如果i小于1(表示没有循环),程序工作正常,但对于(i小于n),其中n大于1则抛出异常

public class Main {

    public static void main(String[] args) {
        final Semaphore sem = new Semaphore(1, true);
        Thread t1 = new Thread("TA") {
            public void run() {
                try {
                    sem.acquire();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("A");
                sem.release();
            }
        };
        Thread t2 = new Thread("TB") {
            public void run() {
                try {
                    sem.acquire();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("B");
                sem.release();
            }
        };
        Thread t3 = new Thread("TC") {
            public void run() {
                try {
                    sem.acquire();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("C");
                sem.release();
            }
        };

        for (int i = 0; i < 10; i++) {
            t1.start();
            t3.start();
            t2.start();
        }

    }
}

1 个答案:

答案 0 :(得分:1)

您不能多次启动一个帖子。因此,在第二个循环中,第二次调用t1.start()时会出现异常。这在javadoc

中说明
  

不止一次启动线程永远不合法。特别是,一旦完成执行,线程可能无法重新启动。

     

抛出:IllegalThreadStateException - 如果线程已经启动。

您可以使用ExecutorService而不是直接操作线程。它看起来像这样:

public static void main(String[] args) throws Exception {
    ExecutorService executor = Executors.newFixedThreadPool(3);
    final Semaphore sem = new Semaphore(1, true);
    Runnable r1 = new Runnable() {
        public void run() {
            try {
                sem.acquire();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("A");
            sem.release();
        }
    };
    Runnable r2 = new Runnable() {
        public void run() {
            try {
                sem.acquire();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("B");
            sem.release();
        }
    };
    Runnable r3 = new Runnable() {
        public void run() {
            try {
                sem.acquire();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("C");
            sem.release();
        }
    };

    for (int i = 0; i < 10; i++) {
        executor.submit(r1);
        executor.submit(r2);
        executor.submit(r3);
    }

    executor.shutdown();
}