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



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

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

for 循环的 i 小于

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();
        }
    }
}

您不能多次启动一个线程。因此,在第二个循环中,第二次调用t1.start()时会出现异常。这在javadoc中有所说明:

多次启动一个线程从来都不合法。特别是,线程在完成执行后可能不会重新启动。

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

您可以使用执行器服务,而不是直接操作线程。它可能看起来像这样:

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();
}

最新更新