三个线程循环顺序打印

简介: 三个线程循环顺序打印

题目描述

三个线程循环打印A、B、C。

解题思路

主要考虑的是线程间同步问题,可以通过线程间等待/通知模型来解决顺序性。

代码实现

方案一

通过thread.join()来控制线程间的顺序性,表示在当前线程中执行另一个线程的thread.join()方法,可以等待另一个线程执行完成之后,当前线程才执行,通过这样的方式来控制两个线程的先后顺序。

/**
 * @author yangnk
 * @desc
 * @date 2023/08/26 15:30
 **/
public class JoinTest {
   
    public static void main(String[] args) throws InterruptedException {
   
        while (true) {
   
            Thread thread1 = new Thread(new MyTask1());
            Thread thread2 = new Thread(new MyTask2(thread1));
            Thread thread3 = new Thread(new MyTask3(thread2));

            thread1.start();
            thread2.start();
            thread3.start();

            thread3.join();
        }

    }
}

class MyTask1 implements Runnable {
   

    @Override
    public void run() {
   
        System.out.println("A");
        try {
   
            Thread.sleep(1000);
        } catch (InterruptedException e) {
   
            e.printStackTrace();
        }
    }
}

class MyTask2 implements Runnable {
   

    Thread thread;

    public MyTask2(Thread thread) {
   
        this.thread = thread;
    }

    @Override
    public void run() {
   
        try {
   
        thread.join();
        System.out.println("B");
        Thread.sleep(1000);
        } catch (InterruptedException e) {
   
            e.printStackTrace();
        }
    }
}

class MyTask3 implements Runnable {
   

    Thread thread;

    public MyTask3(Thread thread) {
   
        this.thread = thread;
    }

    @Override
    public void run() {
   
        try {
   
            thread.join();
            System.out.println("C");
            Thread.sleep(1000);
        } catch (InterruptedException e) {
   
            e.printStackTrace();
        }
    }
}

方案二

通过wait()/notiyf()来实现等待/通知模型,首先线程A获取锁后执行线程A中的打印逻辑,然后线程A调用了lock对象的notifyAll()方法进入释放lock对象,线程B获取到锁后执行线程B中的打印逻辑,然后线程B调用了lock对象的notifyAll()方法进入释放lock对象,线程C获取到锁后执行线程C中的打印逻辑,然后线程C调用了lock对象的notifyAll()方法进入释放lock对象。

适合该场景的等待/通知模型:

(1) 获取全局对象锁。
(2) 如果条件不满足,那么调用对象的wait()方法,此时会释放锁。
(3) 竞争到锁并且条件满足,则执行对应的逻辑。
(4) 释放全局对象锁,其他线程可以开始竞争锁。

synchronized(lock){
    while(条件不满足){
        lock.wait()
    }
    业务逻辑
    lock.notiyfAll()
}

代码实现:

/**
 * @author yangnk
 * @desc
 * @date 2023/08/27 09:40
 **/
public class WaitNotifyTest {
    private static Object lock = new Object();
    private static int num = 0;


    public static void main(String[] args) {
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (lock) {
                    try {
                        while (true) {
                            while (num % 3 != 0) {
                                lock.wait();
                            }
                            Thread.sleep(1000);
                            num++;
                            System.out.println("A");
                            lock.notifyAll();
                        }
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        });

        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (lock) {
                    try {
                        while (true) {
                            while (num % 3 != 1) {
                                lock.wait();
                            }
                            Thread.sleep(1000);
                            num++;
                            System.out.println("B");
                            lock.notifyAll();
                        }
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
        Thread thread3 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (lock) {
                    try {
                        while (true) {
                            while (num % 3 != 2) {
                                lock.wait();
                            }
                            Thread.sleep(1000);
                            num++;
                            System.out.println("C");
                            lock.notifyAll();
                        }
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
        thread1.start();
        thread2.start();
        thread3.start();

    }
}

方案三

通过ReentrantLock+Condition方法来顺序打印,其中Condition可以实现全局锁的多条件控制。

/**
 * @author yangnk
 * @desc
 * @date 2023/08/27 00:16
 **/
public class ReentrantLockLockTest {
    private static ReentrantLock lock = new ReentrantLock();
    private static Condition condition1 = lock.newCondition();
    private static Condition condition2 = lock.newCondition();
    private static Condition condition3 = lock.newCondition();
    private static int num = 0;

    public static void main(String[] args) {
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    lock.lock();
                    try {
                        while (num % 3 != 0) {
                            condition1.await();
                        }
                        Thread.sleep(1000);
                        System.out.println("A");
                        num++;
                        condition2.signal();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        lock.unlock();
                    }
                }
            }
        });
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    lock.lock();
                    try {
                        while (num % 3 != 1) {
                            condition2.await();
                        }
                        Thread.sleep(1000);
                        System.out.println("B");
                        num++;
                        condition3.signal();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        lock.unlock();
                    }
                }
            }
        });
        Thread thread3 = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    lock.lock();
                    try {
                        while (num % 3 != 2) {
                            condition3.await();
                        }
                        Thread.sleep(1000);
                        System.out.println("C");
                        num++;
                        condition1.signal();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        lock.unlock();
                    }
                }
            }
        });

        thread1.start();
        thread2.start();
        thread3.start();
    }
}
目录
相关文章
|
6月前
|
机器学习/深度学习 缓存 Java
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
444 0
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
|
8月前
|
安全 API C++
c++生产者和消费者线程循环
线程安全-生产者消费者模型
79 1
|
6月前
|
消息中间件 移动开发 自然语言处理
多线程知识:三个线程如何交替打印ABC循环100次
synchronized是Java中的一个关键字,用于实现对共享资源的互斥访问。wait和notify是Object类中的两个方法,用于实现线程间的通信。wait方法会让当前线程释放锁,并进入等待状态,直到被其他线程唤醒。notify方法会唤醒一个在同一个锁上等待的线程。
66 1
|
11月前
|
消息中间件 前端开发 NoSQL
蔚来手撕代码题:三个线程循环打印ABC
蔚来手撕代码题:三个线程循环打印ABC
蔚来手撕代码题:三个线程循环打印ABC
|
缓存 NoSQL Redis
什么是自旋锁 自旋锁是指当一个线程尝试获取某个锁时,如果该锁已被其他线程占用,就一直循环检测锁是否被释放,而不是进入线程挂起或睡眠状态。
什么是自旋锁 自旋锁是指当一个线程尝试获取某个锁时,如果该锁已被其他线程占用,就一直循环检测锁是否被释放,而不是进入线程挂起或睡眠状态。
198 0
什么是自旋锁 自旋锁是指当一个线程尝试获取某个锁时,如果该锁已被其他线程占用,就一直循环检测锁是否被释放,而不是进入线程挂起或睡眠状态。
|
Java
Java多线程:用三个线程控制循环输出10次ABC
Java多线程:用三个线程控制循环输出10次ABC
3357 0
java用while循环设计轮询线程的性能问题
java用while循环设计轮询线程的性能问题        轮询线程在开发过程中的应用是比较广泛的,在这我模拟一个场景,有一个队列和轮询线程,主线程往队列中入队消息,轮询线程循环从队列中读取消息并打印消息内容。