java线程间通信[实现不同线程之间的消息传递(通信),生产者和消费者模型]

简介: 线程通信,线程之间的消息传递; 多个线程在操作同一个资源,但对共享资源的操作动作不同;它们共享同一个资源,互为条件,相互依赖,相互通信让任务向前推进。 线程的同步,可以解决并发更新同一个资源,实现线程同步;但不能用来实现线程间的消息传递。 线程通信生产者和消费者和仓库是个典型模型: 生产者:没有生产之前通知消费者等待,生产产品结束之后,马上通知消费者消费 消费者:没有消费之前通知

线程通信,线程之间的消息传递;

多个线程在操作同一个资源,但对共享资源的操作动作不同;它们共享同一个资源,互为条件,相互依赖,相互通信让任务向前推进。

线程的同步,可以解决并发更新同一个资源,实现线程同步;但不能用来实现线程间的消息传递。

线程通信生产者消费者仓库是个典型模型:

生产者:没有生产之前通知消费者等待,生产产品结束之后,马上通知消费者消费

消费者:没有消费之前通知生产者等待,消费产品结束之后,通知生产者继续生产产品以供消费

线程通信:使用java中Object中提供的:

public final void wait();  注:long timeout=0  表示线程一直等待,直到其它线程通知

public final native void wait(long timeout);   线程等待指定毫秒参数的时间

public final void wait(long timeout, int nanos);  线程等待指定毫秒、微妙的时间timeout最大等待时间,以毫秒为单位,nanos额外的时间,在纳秒范围0-999999。

public final native void notify();   唤醒一个处于等待状态的线程

public final native void notifyAll();  唤醒同一个对象上所有调用wait()方法的线程,优先级别高的线程优先运行

这些方法只能在同步方法或者同步代码块中使用,否则会抛出异常。

Exception in thread "Thread-0"java.lang.IllegalMonitorStateException
at java.lang.Object.wait(Native Method)
at java.lang.Object.wait(Object.java:485)
at ca.bb.ShareCached.getShareCachedData(ShareCached.java:41)
  //wait();

以及使用,回调方法,实现线程通信。

此处使用wait()和notify()方法实现。

此例:生产A-D个商品,放入仓库,消费生产。

仓库类:

package ca.bb.threadcommunication;
/**
 * 共享资源缓存和操作类
 * */
public class ShareCached {
	/**产品:此处使用char字符,作为存储共享数据存储类型*/
	private char cache;
	//产品消费标识:线程间通信信号,true未消费(生产),false未生产(消费)
	private boolean flag = false;
	/**
	 * 生产操作(生产者):共享数据添加方法
	 * */
	public synchronized void addShareCachedData(char data){
		//产品未消费,则生产者(生产操作)等待
		if(flag){
			System.out.println("产品未消费,生产者生产操作等待");
			try {
				//生产者等待
				wait();
			} catch (InterruptedException e) {
				System.out.println("Thread Interrupted Exception,"+e.getMessage());
			}
		}
		//产品已消费,则生产者继续生产
		this.cache = data;
		//标记已生产
		flag = true;
		//通知消费者已生产
		notify();
		System.out.println("产品"+data+",已生产,通知消费者消费");
	}
	/**
	 * 消费操作(消费者):共享数据获取方法
	 * */
	public synchronized char getShareCachedData(){
		//产品未生产,则消费者(消费操作)等待
		if(!flag){
			System.out.println("产品未生产,消费者消费操作等待");
			try {
				//消费者等待
				wait();
			} catch (InterruptedException e) {
				System.out.println("Thread Interrupted Exception,"+e.getMessage());
			}
		}
		//标记已消费
		flag = false;
		//通知生产者已消费
		notify();
		System.out.println("产品"+this.cache+",已消费,通知生产者生产");
		//产品已生产,则消费者继续消费
		return this.cache;
	}
}

注:如果此处,使用synchronized方法,则内置锁(对象监视锁)为this,则不能多new不同对象,如果为多线程同步,必定多个线程共享同一个对象监视锁。

使用synchronized(obj)代码块,则1.锁使用涉及当前类对象外的其他类对象;2.锁使用静态当前类对象,确保同一个锁。

生产者线程类:

package ca.bb.threadcommunication;
/**
 * 生产者线程类
 * */
public class Producer extends Thread{
	//共享资源缓存类对象
	private ShareCached sCached;
	/**
	 * 构造加入共享资源操作类
	 * */
	public Producer(ShareCached shareCached){
		this.sCached = shareCached;
	}
	/**
	 * 生产者生产产品,放入共享数据缓存类(仓库)
	 * 生产A-D类型的产品
	 * */
	@Override
	public void run() {
		for(char product = 'A';product <='D';product++){
			try {
				sleep((int)(Math.random()*3000));
			} catch (InterruptedException e) {
				System.out.println("Thread Interrupted Exception,"+e.getMessage());
			}
			//生产产品,放入共享数据缓存类(仓库)
			sCached.addShareCachedData(product);
		}
	}
}
消费者线程类:

package ca.bb.threadcommunication;
/**
 * 消费者线程类
 * */
public class Consumer extends Thread{
	//共享资源缓存类对象
	private ShareCached sCached;
	/**
	 * 构造加入共享资源操作类
	 * */
	public Consumer(ShareCached sharedCached){
		this.sCached = sharedCached;
	}
	/**
	 * 消费者消费产品,获取共享数据缓存类(仓库)
	 * 消费D类型的产品停止消费
	 * */
	@Override
	public void run() {
		char product = '\u0000';
		do {
            try {
                Thread.sleep((int)(Math.random()*3000));
            } catch (InterruptedException e) {
                System.out.println("Consumer thread InterruptedException from run method!");
            }
            //消费,从仓库取走商品
            product = sCached.getShareCachedData();
		} while (product != 'D');
	}
}
线程通信测试类:

package ca.bb.threadcommunication;

public class CPTest {
	public static void main(String[] args) {
		//共享同一个共享资源
		ShareCached shareCached = new ShareCached();
		//启动消费者线程
		new Consumer(shareCached).start();
		//启动生产者线程
		new Producer(shareCached).start();
	}
}
测试结果:

产品未生产,消费者消费操作等待
产品A,已生产,通知消费者消费
产品A,已消费,通知生产者生产
产品B,已生产,通知消费者消费
产品未消费,生产者生产操作等待
产品B,已消费,通知生产者生产
产品C,已生产,通知消费者消费
产品未消费,生产者生产操作等待
产品C,已消费,通知生产者生产
产品D,已生产,通知消费者消费
产品D,已消费,通知生产者生产

目录
相关文章
|
24天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
29 2
|
24天前
|
监控 Java 开发者
Java线程管理:守护线程与本地线程的深入剖析
在Java编程语言中,线程是程序执行的最小单元,它们可以并行执行以提高程序的效率和响应性。Java提供了两种特殊的线程类型:守护线程和本地线程。本文将深入探讨这两种线程的区别,并探讨它们在实际开发中的应用。
30 1
|
2月前
|
并行计算 JavaScript 前端开发
单线程模型
【10月更文挑战第15天】
|
1月前
|
Prometheus 监控 Cloud Native
JAVA线程池监控以及动态调整线程池
【10月更文挑战第22天】在 Java 中,线程池的监控和动态调整是非常重要的,它可以帮助我们更好地管理系统资源,提高应用的性能和稳定性。
107 4
|
2月前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
25 1
|
2月前
|
安全 调度 C#
STA模型、同步上下文和多线程、异步调度
【10月更文挑战第19天】本文介绍了 STA 模型、同步上下文和多线程、异步调度的概念及其优缺点。STA 模型适用于单线程环境,确保资源访问的顺序性;同步上下文和多线程提高了程序的并发性和响应性,但增加了复杂性;异步调度提升了程序的响应性和资源利用率,但也带来了编程复杂性和错误处理的挑战。选择合适的模型需根据具体应用场景和需求进行权衡。
|
Java Linux Windows
JAVA通信编程(五)——串口通讯的补充说明
在《JAVA通讯编程(一)——串口通讯》中讲述了如何采用JAVA进行串口通讯,我们采用的是引入RXTXComm.jar的方式,关于这个我有两点需要说明补充。 首先,现在的笔记本一般都不带串口,需要usb转串口之类的工具才能进行通讯,这样对调试程序非常的不方便,所以在windows操作系统下我们选择采用VSPD(Virtual Serial Port Driver)虚拟串口,VSPD对虚拟串口的序号没有限制,理论上可以创建无数个。
1724 0
|
网络协议 Java Windows
JAVA通信编程(一)——串口通讯
  博主结合实际经验,决定总结下JAVA通讯编程的一些小知识,希望能给给位读者有些帮助。这里的JAVA通讯编程主要是指如何应用JAVA编写串口、TCP以及UDP的通讯程序。本片主要讲述的是串口通讯。
2023 0
|
1天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
3天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。