【CountDownLatch 和 Semaphore】

介绍:

CountDownLatch 和 Semaphore 都是 Java 多线程编程中的同步工具,用于控制线程的执行顺序和对共享资源的访问控制。

CountDownLatch 是一个计数器,允许一个或多个线程等待其他线程完成操作后再执行。在 CountDownLatch 初始化时设置一个计数器,每个线程完成任务后将计数器减 1,当计数器变为 0 时,等待的线程将被唤醒。

Semaphore 是一个信号量,用于控制同时访问共享资源的线程数量。Semaphore 在初始化时设置一个许可证数量,表示可以同时访问资源的线程数量,每个线程访问资源前需要获取一个许可证,访问结束后释放该许可证。

区别:

CountDownLatch 和 Semaphore 在功能上有一些区别,CountDownLatch 用于等待其他线程完成某个操作,而 Semaphore 则用于限制同时访问共享资源的线程数量。

在实现上,CountDownLatch 只能使用一次,即计数器减为 0 后不能再次使用,而 Semaphore 可以重复使用,即许可证数量可以被释放和重新获取。

另外,CountDownLatch 计数器的初始值是固定的,而 Semaphore 的许可证数量可以动态调整。

CountDownLatch


import java.util.concurrent.CountDownLatch;

/**
 * 在这个示例中,我们创建了一个 CountDownLatch 对象,并将其初始化为 5。
 * 然后,我们创建了 5 个线程,并在每个线程中调用 countDown 方法,以表示该线程已完成其任务。
 * 最后,我们调用 await 方法来等待所有线程完成任务,并输出 "All threads completed"。
 *
 * countDownLatch.countDown():表示该线程已完成其任务
 * countDownLatch.await()    :等待所有线程完成任务
 */
public class CountDownLatchExample {
    public static void main(String[] args) throws InterruptedException {
        int numOfThreads = 5;
        CountDownLatch countDownLatch = new CountDownLatch(numOfThreads);

        for (int i = 0; i < numOfThreads; i++) {
            new Thread(() -> {
                System.out.println("Thread " + Thread.currentThread().getId() + " started");
                countDownLatch.countDown();
                System.out.println("Thread " + Thread.currentThread().getId() + " completed");
            }).start();
        }

        countDownLatch.await();
        System.out.println("All threads completed");
    }
}

Semaphore


import java.util.concurrent.Semaphore;

/**
 * 在这个示例中,我们创建了一个 Semaphore 对象,并将其初始化为 3。然后,我们创建了 5 个线程,
 * 并在每个线程中调用 acquire 方法来获取许可证。如果许可证数量已达到 3 个,则线程将被阻塞直到有一个许可证可用.
 * 当线程完成其任务后,它将调用 release 方法来释放许可证,并使其可用于其他线程。
 *
 * 在这个示例中,我们使用 Thread.sleep 来模拟每个线程完成其任务所需的时间。
 *
 * semaphore.acquire():获取许可证
 * semaphore.release():释放许可证
 */
public class SemaphoreExample {
    public static void main(String[] args) {
        int numOfPermits = 3;
        Semaphore semaphore = new Semaphore(numOfPermits);

        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                try {
                    semaphore.acquire();
                    System.out.println("Thread " + Thread.currentThread().getId() + " acquired a permit");
                    Thread.sleep(1000);
                    semaphore.release();
                    System.out.println("Thread " + Thread.currentThread().getId() + " released a permit");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }
}

你可能感兴趣的:(记录,Java,多线程,java,jvm,开发语言)