Java多个线程顺序循环执行的几种实现方式

ReentrantLock+Condition方式实现

public class ReentrantLock_Impl {

    private static final String FLAG_THREAD_1 = "ReentrantLock_Thread1";
    private static final String FLAG_THREAD_2 = "ReentrantLock_Thread2";
    private static final String FLAG_THREAD_3 = "ReentrantLock_Thread3";

    public static void main(String[] args) throws InterruptedException {
        //ReentrantLock构造方法传入true为公平锁 false为非公平锁
        ReentrantLock lock = new ReentrantLock(true);
        Condition condition1 = lock.newCondition();
        Condition condition2 = lock.newCondition();
        Condition condition3 = lock.newCondition();
        FairRunnable runnable = new FairRunnable(lock, condition1, condition2, condition3);

        new Thread(runnable, FLAG_THREAD_1).start();
        new Thread(runnable, FLAG_THREAD_2).start();
        new Thread(runnable, FLAG_THREAD_3).start();

    }

    static class FairRunnable implements Runnable {
        private ReentrantLock lock;
        private Condition condition1;
        private Condition condition2;
        private Condition condition3;

        public FairRunnable(ReentrantLock lock, Condition condition1, Condition condition2, Condition condition3) {
            this.lock = lock;
            this.condition1 = condition1;
            this.condition2 = condition2;
            this.condition3 = condition3;
        }

        @Override
        public void run() {
            while (true) {
                try {
                    lock.lock();
                    System.out.println(Thread.currentThread().getName() + "开始执行");
                    Thread.sleep(1000);
                    switch (Thread.currentThread().getName()) {
                        case FLAG_THREAD_1:
                            //唤醒线程2 自身线程挂起阻塞
                            condition2.signal();
                            condition1.await();
                            break;
                        case FLAG_THREAD_2:
                            //唤醒线程3 自身线程挂起阻塞
                            condition3.signal();
                            condition2.await();
                            break;
                        case FLAG_THREAD_3:
                            //唤醒线程1 自身线程挂起阻塞
                            condition1.signal();
                            condition3.await();
                            break;
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                    return;
                } finally {
                    lock.unlock();
                }
            }
        }
    }
}

执行结果:

ReentrantLock_Thread1开始执行
ReentrantLock_Thread2开始执行
ReentrantLock_Thread3开始执行
ReentrantLock_Thread1开始执行
ReentrantLock_Thread2开始执行
ReentrantLock_Thread3开始执行
......

Synchronized+wait/notifyAll方式实现

public class Synchronized_Impl {
    private static final String FLAG_THREAD_1 = "Synchronized_Thread1";
    private static final String FLAG_THREAD_2 = "Synchronized_Thread2";
    private static final String FLAG_THREAD_3 = "Synchronized_Thread3";

    public static void main(String[] args) {
        FairRunnable runnable = new FairRunnable();
        new Thread(runnable, FLAG_THREAD_1).start();
        new Thread(runnable, FLAG_THREAD_2).start();
        new Thread(runnable, FLAG_THREAD_3).start();
    }

    static class FairRunnable implements Runnable {
        private volatile static int flag = 1;
        private final Object object = new Object();

        @Override
        public void run() {
            while (true) {
                synchronized (object) {
                    //如果当前情况是:线程1&flag!=1、线程2&flag!=2、线程3&flag!=3 那当前线程通过object.wait挂起
                    switch (Thread.currentThread().getName()) {
                        case FLAG_THREAD_1:
                            while (flag != 1) {
                                try {
                                    object.wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }
                            break;
                        case FLAG_THREAD_2:
                            while (flag != 2) {
                                try {
                                    object.wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }
                            break;
                        case FLAG_THREAD_3:
                            while (flag != 3) {
                                try {
                                    object.wait();
                                } catch (InterruptedException e) {
                                    e.printStackTrace();
                                }
                            }
                            break;
                    }
                    //线程任务开始执行
                    System.out.println(Thread.currentThread().getName() + "开始执行");
                    try {
                        //模拟线程任务执行
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                        break;
                    }
                    switch (Thread.currentThread().getName()) {
                        case FLAG_THREAD_1:
                            //接下来该去执行线程2
                            flag = 2;
                            break;
                        case FLAG_THREAD_2:
                            //接下来该去执行线程3
                            flag = 3;
                            break;
                        case FLAG_THREAD_3:
                            //接下来该去执行线程1
                            flag = 1;
                            break;
                    }
                    //唤醒所有线程
                    object.notifyAll();
                }
            }
        }
    }
}

执行结果:

Synchronized_Thread1开始执行
Synchronized_Thread2开始执行
Synchronized_Thread3开始执行
Synchronized_Thread1开始执行
Synchronized_Thread2开始执行
Synchronized_Thread3开始执行
......

两者对比

使用ReentrantLock+Condition可以更准确的控制唤醒哪一个线程;而Synchronized+wait/notifyAll的方式可能会出现获取锁的线程并不是目标线程,此时获取锁的线程会重新挂起,直到获取锁的线程即是目标线程为止。比如上面代码中:线程1执行完后调用notifyAll(), 此时线程2、线程3都会被唤醒并尝试获取锁,如果此时线程3获得锁,那么线程2还要继续等待,线程3执行时发现并不是要执行的目标线程,那么线程3会调用wait()挂起,此时entrySet中只有线程2了,线程2也是要执行的目标线程,此时才会去执行线程2,整个过程多了一次线程3的获取与释放锁,原因就是wait/notifyAll并不能像ReentrantLock+Condition一样能精确地唤醒某个线程。

  • 3
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Java中,可以通过多线程执行for循环实现并发执行任务的效果。有几种常见的方法可以实现这个目标。 引用中的示例代码使用了ThreadUtil.execAsync()方法来创建并启动子线程,每个子线程执行一次for循环中的任务。这种方式是将每个任务交给线程池来执行,主线程与子线程之间没有先后顺序。 引用中的示例代码使用了CountDownLatch来控制主线程等待所有子线程执行完成。在每个子线程的任务执行完毕后,都会调用countDownLatch.countDown()来将计数器减1。主线程通过调用countDownLatch.await()方法来阻塞等待,直到计数器减到0,即所有子线程执行完成后才会继续执行。 引用中的示例代码也使用了CountDownLatch来实现线程等待所有子线程执行完成。在每个子线程的任务执行完毕后,都会调用countDownLatch.countDown()来将计数器减1。主线程通过调用countDownLatch.await()方法来阻塞等待,直到计数器减到0,即所有子线程执行完成后才会继续执行。 所以,以上这些方法都可以实现Java线程执行for循环的效果,具体选择哪种方法取决于实际需求和代码结构。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [JAVA线程处理for循环数据](https://blog.csdn.net/m0_50932526/article/details/130204863)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

_小马快跑_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值