六种java的多线程设计模式详解和代码举例
yund56 2025-07-28 23:32 28 浏览
java的多线程处理,有哪些模式可以使用呢,如何使用呢。本文列举了六种多线程设计模式供大家参考。
1.生产者-消费者模式
设计理念:
生产者-消费者模式通过协调两个线程(生产者和消费者)来处理数据,生产者生成数据并将其放入队列,消费者从队列中取出数据进行处理。这种模式可以有效地解耦数据的生成和消费过程。
举个代码栗子如下:
import java.util.LinkedList;
import java.util.Queue;
class Producer implements Runnable {
private Queue<Integer> queue; // 共享队列
private final int bound; // 队列容量上限
Producer(Queue<Integer> q, int bound) {
this.queue = q;
this.bound = bound;
}
public void run() {
while (true) {
int item = produce(); // 生产数据
// 如果队列已满,等待消费者消费
if (queue.size() == bound) {
System.out.println("Queue is full. Waiting for consumer");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
queue.add(item); // 将数据放入队列
System.out.println("Produced " + item);
}
}
int produce() {
return (int) (Math.random() * 100); // 模拟生产数据
}
}
class Consumer implements Runnable {
private Queue<Integer> queue;
Consumer(Queue<Integer> q) {
this.queue = q;
}
public void run() {
while (true) {
// 如果队列为空,等待生产者生产
if (queue.isEmpty()) {
System.out.println("Queue is empty. Waiting for producer");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
int item = queue.remove(); // 从队列中取出数据
consume(item); // 消费数据
System.out.println("Consumed " + item);
}
}
void consume(int item) {
// 消费数据的具体实现
}
}
public class ProducerConsumerExample {
public static void main(String[] args) {
Queue<Integer> q = new LinkedList<>();
Thread t1 = new Thread(new Producer(q, 5)); // 创建生产者线程
Thread t2 = new Thread(new Consumer(q)); // 创建消费者线程
t1.start();
t2.start();
}
}可能出现的问题:
o 死锁: 如果生产者和消费者之间的同步机制不当,可能会导致死锁。
o 资源浪费: 如果队列大小设置不当,可能会导致频繁的线程阻塞和唤醒,造成资源浪费。
o 饥饿或堆积: 如果消费者线程比生产者线程多,生产者可能会饿死;反之则可能出现堆积。
回避手段:
o 使用合适的同步机制: 使用 BlockingQueue 等线程安全的数据结构,它们提供了必要的同步机制。
o 合理设置队列大小: 根据实际需求合理设置队列大小,避免资源浪费。
o 平衡生产者和消费者数量: 根据任务特性和系统资源合理分配生产者和消费者的数量。
2.线程池模式
设计理念:
线程池模式通过复用一组线程来执行任务,减少了频繁创建和销毁线程的开销,提高了效率。
举个代码栗子:
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
class Task implements Runnable {
private final String name;
Task(String name) {
this.name = name;
}
public void run() {
System.out.println("Task " + name + " is running");
}
}
public class ThreadPoolExample {
public static void main(String[] args) {
ExecutorService executor = Executors.newFixedThreadPool(5); // 创建固定大小的线程池
for (int i = 0; i < 10; i++) {
executor.execute(new Task("" + i)); // 提交任务到线程池
}
executor.shutdown(); // 关闭线程池
}
}可能出现的问题:
o 线程池参数设置不当: 线程池大小设置不合适可能导致资源浪费或任务执行延迟。
o 任务执行顺序不确定: 线程池中的任务执行顺序可能不是提交的顺序。
回避手段:
o 合理配置线程池参数: 根据任务特性和系统资源合理配置线程池的大小、最大任务队列长度等参数。
o 使用优先队列: 如果需要任务执行顺序,可以使用优先队列来管理任务。
3.Futures和Promises模式
设计理念:
Futures和Promises模式允许异步执行任务并在未来某个时间点获取结果,适用于需要非阻塞操作的场景。
举个代码栗子:
import java.util.concurrent.*;
class CallableTask implements Callable<String> {
private final String name;
CallableTask(String name) {
this.name = name;
}
@Override
public String call() throws Exception {
Thread.sleep(1000); // 模拟任务执行时间
return "Result of " + name;
}
}
public class FuturesAndPromisesExample {
public static void main(String[] args) throws ExecutionException, InterruptedException {
ExecutorService executor = Executors.newFixedThreadPool(2); // 创建线程池
Future<String> future = executor.submit(new CallableTask("Task 1")); // 提交Callable任务
System.out.println("Future result: " + future.get()); // 获取结果
executor.shutdown(); // 关闭线程池
}
}可能出现的问题:
o 异常处理困难: 异步任务的异常可能不容易被捕获和处理。
o 结果获取阻塞: Future.get() 方法可能会阻塞主线程,直到异步任务完成。
回避手段:
o 适当的异常处理: 使用 try-catch 块来捕获和处理异步任务的异常。
o 非阻塞结果获取: 使用 Future 的 isDone() 方法检查任务是否完成,或者使用回调函数来处理结果。
4.监视器模式
设计理念:
监视器模式是一种同步机制,它允许多个线程访问共享资源,同时确保同一时间只有一个线程可以访问共享资源。监视器模式通常通过`synchronized`关键字实现,它可以保护方法或代码块,确保线程安全。
举个代码栗子:
class SharedObject {
private int data;
public synchronized void setData(int data) { // 同步方法保护共享资源
this.data = data;
}
public synchronized int getData() { // 同步方法保护共享资源
return data;
}
}
class Writer implements Runnable {
private final SharedObject sharedObject;
Writer(SharedObject sharedObject) {
this.sharedObject = sharedObject;
}
public void run() {
sharedObject.setData(20); // 写入数据
}
}
class Reader implements Runnable {
private final SharedObject sharedObject;
Reader(SharedObject sharedObject) {
this.sharedObject = sharedObject;
}
public void run() {
System.out.println("Data: " + sharedObject.getData()); // 读取数据
}
}
public class MonitorObjectExample {
public static void main(String[] args) {
SharedObject sharedObject = new SharedObject();
Thread writer = new Thread(new Writer(sharedObject)); // 创建写线程
Thread reader = new Thread(new Reader(sharedObject)); // 创建读线程
writer.start();
reader.start();
}
}在这个示例中,`SharedObject`类中的`setData`和`getData`方法都被声明为`synchronized`,这意味着同一时间只有一个线程可以执行这些方法中的任何一个。这确保了当一个线程正在修改数据时,其他线程不能读取或修改数据,从而避免了数据不一致的问题。
可能出现的问题:
o 死锁: 如果不正确地使用 synchronized 关键字,可能会导致死锁。
o 性能问题: 过度同步可能会导致性能下降。
回避手段:
o 最小化同步范围: 只在必要的时候同步代码块,而不是整个方法。
o 避免在同步块中执行长时间操作: 例如,不要在同步块中进行I/O操作。
5.屏障模式
设计理念:
屏障模式(CyclicBarrier)允许一组线程相互等待,直到所有线程都到达一个公共屏障点,然后可以选择执行一个共同的任务(如计数器重置)后继续执行。
举个代码栗子:
import java.util.concurrent.CyclicBarrier;
class BarrierTask implements Runnable {
private final CyclicBarrier barrier;
BarrierTask(CyclicBarrier barrier) {
this.barrier = barrier;
}
public void run() {
try {
System.out.println("Before barrier");
barrier.await(); // 等待其他线程到达屏障
System.out.println("After barrier");
} catch (Exception e) {
e.printStackTrace();
}
}
}
public class BarrierExample {
public static void main(String[] args) {
CyclicBarrier barrier = new CyclicBarrier(2, () -> System.out.println("All threads have reached the barrier")); // 创建屏障
Thread t1 = new Thread(new BarrierTask(barrier)); // 创建线程
Thread t2 = new Thread(new BarrierTask(barrier)); // 创建线程
t1.start();
t2.start();
}
}可能出现的问题:
o 屏障释放问题: 如果屏障没有被正确释放,可能会导致线程永远等待。
o 线程中断处理: 在等待屏障释放时,线程中断可能没有被正确处理。
回避手段:
o 确保屏障被释放: 在所有线程到达屏障后,确保执行屏障的 await() 方法。
o 正确处理中断: 在 await() 方法中捕获 InterruptedException ,并根据需要恢复线程状态或重新中断。
6.读写锁模式
设计理念:
读写锁模式(ReentrantReadWriteLock)允许多个读操作同时进行,而写操作是互斥的,这样可以提高读操作的并发性,特别是在读多写少的场景下。
举个代码栗子:
import java.util.concurrent.locks.ReentrantReadWriteLock;
class SharedData {
private ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock(); // 读写锁
private int data;
void writeData(int data) {
rwLock.writeLock().lock(); // 获取写锁
try {
this.data = data;
} finally {
rwLock.writeLock().unlock(); // 释放写锁
}
}
int readData() {
rwLock.readLock().lock(); // 获取读锁
try {
return data;
} finally {
rwLock.readLock().unlock(); // 释放读锁
}
}
}
class Writer implements Runnable {
private final SharedData sharedData;
Writer(SharedData sharedData) {
this.sharedData = sharedData;
}
public void run() {
sharedData.writeData(10); // 写入数据
}
}
class Reader implements Runnable {
private final SharedData sharedData;
Reader(SharedData sharedData) {
this.sharedData = sharedData;
}
public void run() {
System.out.println("Data: " + sharedData.readData()); // 读取数据
}
}
public class ReadWriteLockExample {
public static void main(String[] args) {
SharedData sharedData = new SharedData();
Thread writer = new Thread(new Writer(sharedData)); // 创建写线程
Thread reader = new Thread(new Reader(sharedData)); // 创建读线程
writer.start();
reader.start();
}
}可能出现的问题:
o 写饥饿: 如果读操作非常频繁,写操作可能会被饿死。
o 死锁: 如果一个线程同时持有读锁和写锁,可能会导致死锁。
回避手段:
o 限制读锁的持有时间: 尽量减少读锁的持有时间,避免长时间占用读锁。
o 避免锁升级: 不要从读锁升级到写锁,因为这可能导致死锁。
以上我们提供了对多线程各种模式的解释和代码举例,也包括它们的设计理念和使用时需要注意的问题。
希望这些信息能帮助你更好地应用这些多线程设计模式。
今日份的程序员鼓励师,加油哦
相关推荐
- SM小分队Girls on Top,女神战队少了f(x)?
-
这次由SM娱乐公司在冬季即将开演的smtown里,将公司的所有女团成员集结成了一个小分队project。第一位这是全面ACE的大姐成员权宝儿(BoA),出道二十年,在日本单人销量过千万,韩国国内200...
- 韩国女团 aespa 首场 VR 演唱会或暗示 Quest 3 将于 10 月推出
-
AmazeVR宣布将在十月份举办一场现场VR音乐会,观众将佩戴MetaQuest3进行体验。韩国女团aespa于2020年11月出道,此后在日本推出了三张金唱片,在韩国推出了...
- 韩网热议!女团aespa成员Giselle在长腿爱豆中真的是legend
-
身高163的Giselle,长腿傲人,身材比例绝了...
- 假唱而被骂爆的女团:IVE、NewJeans、aespa上榜
-
在韩国,其实K-pop偶像并不被认为是真正的歌手,因为偶像们必须兼备舞蹈能力、也经常透过对嘴来完成舞台。由于科技的日渐发达,也有许多网友会利用消音软体来验证K-pop偶像到底有没有开麦唱歌,导致假唱这...
- 新女团Aespa登时尚大片 四个少女四种style
-
来源:环球网
- 韩国女团aespa新歌MV曝光 画面梦幻造型超美
-
12月20日,韩国女团aespa翻唱曲《DreamsComeTrue》MV公开,视频中,她们的造型超美!WINTER背后长出一双梦幻般的翅膀。柳智敏笑容甜美。宁艺卓皮肤白皙。GISELLE五官精致...
- 女网友向拳头维权,自称是萨勒芬妮的原型?某韩国女团抄袭KDA
-
女英雄萨勒芬妮(Seraphine)是拳头在2020年推出的第五位新英雄,在还没有正式上线时就备受lsp玩家的关注,因为她实在是太可爱了。和其他新英雄不同的是,萨勒芬妮在没上线时就被拳头当成虚拟偶像来...
- 人气TOP女团是?INS粉丝数见分晓;TWICE成员为何在演唱会落泪?
-
现在的人气TOP女团是?INS粉丝数见分晓!现在爱豆和粉丝之间的交流方法变得多种多样,但是Instagram依然是主要的交流手段。很多粉丝根据粉丝数评价偶像的人气,拥有数百、数千万粉丝的组合作为全球偶...
- 韩国女团MVaespa Drama MV_韩国女团穿超短裙子跳舞
-
WelcometoDrama.Pleasefollow4ruleswhilewatchingtheDrama.·1)Lookbackimmediatelywhenyoufe...
- aespa师妹团今年将出道! SM职员亲口曝「新女团风格、人数」
-
记者刘宛欣/综合报导南韩造星工厂SM娱乐曾打造出东方神起、SUPERJUNIOR、少女时代、SHINee、EXO等传奇团体,近年推出的aespa、RIIZE更是双双成为新生代一线团体,深受大众与粉丝...
- 南韩最活跃的女团aespa,新专辑《Girls》即将发布,盘点昔日经典
-
女团aespa歌曲盘点,新专辑《Girls》即将发布,期待大火。明天也就是2022年的7月8号,aespa新专辑《Girls》即将发行。这是继首张专辑《Savage》之后,时隔19个月的第二张专辑,这...
- 章泽天女团aespa出席戛纳晚宴 宋康昊携新片亮相
-
搜狐娱乐讯(山今/文玄反影/图科明/视频)法国时间5月23日晚,女团aespa、宋康昊、章泽天等明星亮相戛纳晚宴。章泽天身姿优越。章泽天肩颈线优越。章泽天双臂纤细。章泽天仪态端正。女团aespa亮...
- Aespa舞台暴露身高比例,宁艺卓脸大,柳智敏有“TOP”相
-
作为SM公司最新女团aespa,初舞台《BlackMamba》公开,在初舞台里,看得出来SM公司是下了大功夫的,虽然之前SM公司新出的女团都有很长的先导片,但是aespa显然是有“特殊待遇”。运用了...
- AESPA女团成员柳智敏karina大美女
-
真队内速度最快最火达成队内首个且唯一两百万点赞五代男女团中输断层第一(图转自微博)...
- 对来学校演出的女团成员语言性骚扰?韩国这所男高的学生恶心透了
-
哕了……本月4日,景福男子高中相关人士称已经找到了在SNS中上传对aespa成员进行性骚扰文章的学生,并开始着手调查。2日,SM娱乐创始人李秀满的母校——景福高中迎来了建校101周年庆典活动。当天,S...
- 一周热门
- 最近发表
-
- SM小分队Girls on Top,女神战队少了f(x)?
- 韩国女团 aespa 首场 VR 演唱会或暗示 Quest 3 将于 10 月推出
- 韩网热议!女团aespa成员Giselle在长腿爱豆中真的是legend
- 假唱而被骂爆的女团:IVE、NewJeans、aespa上榜
- 新女团Aespa登时尚大片 四个少女四种style
- 韩国女团aespa新歌MV曝光 画面梦幻造型超美
- 女网友向拳头维权,自称是萨勒芬妮的原型?某韩国女团抄袭KDA
- 人气TOP女团是?INS粉丝数见分晓;TWICE成员为何在演唱会落泪?
- 韩国女团MVaespa Drama MV_韩国女团穿超短裙子跳舞
- aespa师妹团今年将出道! SM职员亲口曝「新女团风格、人数」
- 标签列表
-
- filter函数js (37)
- filter函数excel用不了 (73)
- 商城开发 (40)
- 影视网站免费源码最新版 (57)
- 影视资源api接口 (46)
- 网站留言板代码大全 (56)
- java版软件下载 (52)
- java教材电子课本下载 (48)
- 0基础编程从什么开始学 (50)
- java是用来干嘛的 (51)
- it入门应该学什么 (55)
- java线上课程 (55)
- 学java的软件叫什么软件 (38)
- 程序开发软件有哪些 (53)
- 软件培训 (59)
- 机器人编程代码大全 (50)
- 少儿编程教程免费 (45)
- 新代系统编程教学 (61)
- 共创世界编程网站 (38)
- 亲测源码 (36)
- 三角函数积分公式表 (35)
- 函数的表示方法 (34)
- 表格乘法的公式怎么设置 (34)
- sumif函数的例子 (34)
- 图片素材 (36)
