三种方法教你实现多线程交替打印ABC,干货满满!

简介: 本文介绍了多线程编程中的经典问题——多线程交替打印ABC。通过三种方法实现:使用`wait()`和`notify()`、`ReentrantLock`与`Condition`、以及`Semaphore`。每种方法详细讲解了实现步骤和代码示例,帮助读者理解和掌握线程间的同步与互斥,有效解决并发问题。适合不同层次的开发者学习参考。



Hello,大家好,我是你们活力四射的技术小伙伴小米!今天我们继续实战算法系列的主题,来聊一聊多线程编程中的一个经典问题——多线程交替打印ABC。

问题背景

假设有三个线程,分别打印字母A、B、C。我们需要让这三个线程交替运行,按顺序打印出“ABCABCABC...”,直到打印一定次数或者满足某个条件。如何通过多线程的协调实现这个任务呢?这听起来简单,实际涉及到线程之间的同步和互斥,是我们学习多线程编程的一个很好的练习。

多线程编程的挑战

在多线程编程中,最大的问题就是如何控制多个线程的执行顺序。线程是并发执行的,也就是说它们的执行顺序在没有约束的情况下是不可预知的。为了确保多个线程按照我们期望的顺序执行,就需要使用一些同步机制,比如锁、条件变量、信号量等。

接下来,我会带大家用三种方式来实现这个任务。我们会分别使用Objectwait()notify()方法、ReentrantLockCondition、以及信号量来实现多线程交替打印ABC。

方案一:使用wait()和notify()

首先,最常用的一种方法是利用Java中Object类自带的wait()notify()方法来实现线程之间的同步。每个线程在完成它的打印任务后,通知下一个线程开始执行。

实现步骤

  1. 定义一个共享对象用来同步。
  2. 使用wait()让线程进入等待状态。
  3. 使用notify()唤醒下一个线程。

实现代码

运行结果

在这个实现中,我们使用了wait()notifyAll()方法来控制线程的执行顺序。每个线程在不该自己执行的时候调用wait()方法进入等待状态,直到被下一个线程通过notifyAll()方法唤醒。

方案二:使用ReentrantLock和Condition

第二种方法是使用ReentrantLockCondition类。ReentrantLock是Java中更高级的锁机制,可以控制多个条件变量,而Condition则可以用来替代wait()notify()的功能。

实现步骤

  1. 定义一个ReentrantLock和多个Condition
  2. 每个线程等待相应的Condition,当符合条件时打印字符并唤醒下一个线程。

实现代码

运行结果

在这个实现中,我们使用ReentrantLockCondition来替代了wait()notify()。通过lock来确保线程安全,通过Condition来控制每个线程的执行顺序。

方案三:使用信号量

最后一种方法是使用Semaphore类。Semaphore是一个计数信号量,可以控制多个线程之间的协调。这里,我们使用三个信号量,分别控制线程A、B、C的执行。

实现步骤

定义三个信号量semA、semB、semC

每个线程在自己的信号量上等待,打印完成后释放下一个线程的信号量。

实现代码

运行结果

在这个实现中,信号量控制了线程的执行顺序。初始时,semA的计数为1,而semBsemC为0,这保证了线程A先运行,然后通过释放信号量来依次唤醒线程B和C。

END

今天我们学习了三种实现多线程交替打印ABC的方法:使用wait()notify(),使用ReentrantLockCondition,以及使用信号量。通过这些方法,我们可以有效地控制线程之间的同步与互斥,解决复杂的并发问题。

这三种方法各有优劣。wait()notify()较为基础,适合简单的场景;ReentrantLockCondition提供了更细粒度的控制,适合复杂的并发场景;而Semaphore则是一种经典的计数信号量机制,在某些场景下显得更加直观和高效。

希望这篇文章对大家理解多线程编程有所帮助!如果大家有其他问题或者想要了解更多,欢迎在评论区讨论哦!咱们下次见啦~

我是小米,一个喜欢分享技术的29岁程序员。如果你喜欢我的文章,欢迎关注我的微信公众号软件求生,获取更多技术干货!

相关文章
|
4月前
|
Java 调度
Java并发基础-线程简介(状态、常用方法)
Java并发基础-线程简介(状态、常用方法)
34 0
|
1月前
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
|
1月前
|
Java Spring
运行@Async注解的方法的线程池
自定义@Async注解线程池
79 3
|
2月前
|
安全 Java API
|
2月前
|
Java
java开启线程的四种方法
这篇文章介绍了Java中开启线程的四种方法,包括继承Thread类、实现Runnable接口、实现Callable接口和创建线程池,每种方法都提供了代码实现和测试结果。
java开启线程的四种方法
【多线程面试题 二】、 说说Thread类的常用方法
Thread类的常用方法包括构造方法(如Thread()、Thread(Runnable target)等)、静态方法(如currentThread()、sleep(long millis)、yield()等)和实例方法(如getId()、getName()、interrupt()、join()等),用于线程的创建、控制和管理。
|
2月前
|
Dart API C语言
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作
|
2月前
|
Java UED
基于SpringBoot自定义线程池实现多线程执行方法,以及多线程之间的协调和同步
这篇文章介绍了在SpringBoot项目中如何自定义线程池来实现多线程执行方法,并探讨了多线程之间的协调和同步问题,提供了相关的示例代码。
358 0
|
4月前
|
Java 开发者
线程通信的方法和实现技巧详解
线程通信的方法和实现技巧详解
|
3月前
|
Java
不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor
不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor