线程的通讯

 线程通讯: 一个线程完成了自己的任务时,要通知另外一个线程去完成另外一个任务.
 生产者与消费者

wait():  等待   如果线程执行了wait方法,那么该线程会进入等待的状态,等待状态下的线程必须要被其他线程调用notify方法才能唤醒。
notify(): 唤醒    唤醒线程池等待线程其中的一个。
notifyAll() : 唤醒线程池所有等待 线程。

wait与notify方法要注意的事项:
1. wait方法与notify方法是属于Object对象 的。 因为锁对象可以是任意的
2. wait方法与notify方法必须要在同步代码块或者是同步函数中才能 使用。 因为这两个方法是由锁对象调用的,只有存在同步代码块才存在锁对象
3. wait方法与notify方法必需要由锁对象调用。因为是以锁对象为标识符建立线程池的

执行了wait方法后会释放锁对象

问题一:出现了线程安全问题。 价格错乱了... 

 

package cn.hcx.thread;

class Product1{
	String name;
	double price;
}


class Producer1 extends Thread{
	 
	Product p;
	
	public Producer1(Product p){
		this.p = p;
	}
	
	@Override
	public void run() {
		int i = 0;
		while(true){
			if(i%2==0){
				p.name = "苹果";
				try {
					Thread.sleep(10);
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
				
				p.price = 6.5;
				
			}else{
				p.name = "香蕉";
				p.price = 2.0;
				
			}
			System.out.println("生产者生产出了"+p.name+" 价格是"+p.price);
			i++;
		}
	}
}


class Customer1 extends Thread{
	Product p;
	
	public Customer1(Product p){
		this.p = p;
	}
	@Override
	public void run() {
		
		while(true){
			System.out.println("消费者消费了"+p.name+"  价格是"+p.price);
		}
	}
}

public class Demo51 {
	public static void main(String[] args) {
		Product p = new Product();
		
		Producer1 producer = new Producer1(p);
		Customer1 customer = new Customer1(p);
		producer.start();
		customer.start();
		
	}
}

 

 

 

加锁:并且实现生产一个,消费一个
 

 

class Product{
	
	String name;  //名字
	
	double price;  //价格
	
	boolean flag = false; //产品是否生产完毕的标识,默认情况是没有生产完成。
	
}

//生产者
class Producer extends Thread{
	
	Product  p ;  	//产品
	
	public Producer(Product p) {
		this.p  = p ;
	}
	
	
	
	@Override
	public void run() {
		int i = 0 ; 
		while(true){
		 synchronized (p) {
			if(p.flag==false){
				 if(i%2==0){
					 p.name = "苹果";
					 p.price = 6.5;
				 }else{
					 p.name="香蕉";
					 p.price = 2.0;
				 }
				 System.out.println("生产者生产出了:"+ p.name+" 价格是:"+ p.price);
				 p.flag = true;
				 i++;
				 p.notifyAll(); //唤醒消费者去消费
			}else{
				//已经生产 完毕,等待消费者先去消费
				try {
					p.wait();   //生产者等待
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
			 
		}	
	  }	
	}
}


//消费者
class Customer extends Thread{
	
	Product p; 
	
	public  Customer(Product p) {
		this.p = p;
	}
	
	
	@Override
	public void run() {
		while(true){
			synchronized (p) {	
				if(p.flag==true){  //产品已经生产完毕
					System.out.println("消费者消费了"+p.name+" 价格:"+ p.price);
					p.flag = false; 
					p.notifyAll(); // 唤醒生产者去生产
				}else{
					//产品还没有生产,应该 等待生产者先生产。
					try {
						p.wait(); //消费者也等待了...
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}	
	}
}

public class Demo5 {
	
	public static void main(String[] args) {
		Product p = new Product();  //产品
		//创建生产对象
		Producer producer = new Producer(p);
		//创建消费者
		Customer customer = new Customer(p);
		//调用start方法开启线程
		producer.start();
		customer.start();
		
		
	}
	
}

 

 

 

 

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值