Java多线程之消费者生产者模式

45 篇文章 0 订阅
11 篇文章 0 订阅

这个实例应该是学习线程的一个经典例子,生产者和消费者模式。代码写的很好,详细请看内容。

/*@author shijin  
* 生产者与消费者模型中,要保证以下几点:  
* 1 同一时间内只能有一个生产者生产     生产方法加锁sychronized  
* 2 同一时间内只能有一个消费者消费     消费方法加锁sychronized  
* 3 生产者生产的同时消费者不能消费     生产方法加锁sychronized  
* 4 消费者消费的同时生产者不能生产     消费方法加锁sychronized  
* 5 共享空间空时消费者不能继续消费     消费前循环判断是否为空,空的话将该线程wait,释放锁允许其他同步方法执行  
* 6 共享空间满时生产者不能继续生产     生产前循环判断是否为满,满的话将该线程wait,释放锁允许其他同步方法执行     
*/ 
 
//主类  
class  ProducerConsumer  
{  
    public static void main(String[] args)   
    {  
        StackBasket s = new StackBasket();  
        Producer p = new Producer(s);  
        Consumer c = new Consumer(s);  
        Thread tp = new Thread(p);  
        Thread tc = new Thread(c);  
        tp.start();  
        tc.start();  
    }  
}  
 
//  
class Mantou  
{  
    private int id;  
      
    Mantou(int id){  
        this.id = id;  
    }  
 
    public String toString(){  
        return "Mantou :" + id;  
    }  
}  
 
//共享栈空间  
class StackBasket  
{  
    Mantou sm[] = new Mantou[6];  
    int index = 0;  
      
    /**   
    * show 生产方法.  
    * show 该方法为同步方法,持有方法锁;  
    * show 首先循环判断满否,满的话使该线程等待,释放同步方法锁,允许消费;  
    * show 当不满时首先唤醒正在等待的消费方法,但是也只能让其进入就绪状态,  
    * show 等生产结束释放同步方法锁后消费才能持有该锁进行消费  
    * @param m 元素  
    * @return 没有返回值   
    */   
 
    public synchronized void push(Mantou m){  
        try{  
            while(index == sm.length){  
                System.out.println("!!!!!!!!!生产满了!!!!!!!!!");  
                this.wait();  
            }  
            this.notify();  
        }catch(InterruptedException e){  
            e.printStackTrace();  
        }catch(IllegalMonitorStateException e){  
            e.printStackTrace();  
        }  
          
        sm[index] = m;  
        index++;  
        System.out.println("生产了:" + m + " 共" + index + "个馒头");  
    }  
 
    /**   
    * show 消费方法  
    * show 该方法为同步方法,持有方法锁  
    * show 首先循环判断空否,空的话使该线程等待,释放同步方法锁,允许生产;  
    * show 当不空时首先唤醒正在等待的生产方法,但是也只能让其进入就绪状态  
    * show 等消费结束释放同步方法锁后生产才能持有该锁进行生产  
    * @param b true 表示显示,false 表示隐藏   
    * @return 没有返回值   
    */   
    public synchronized Mantou pop(){  
        try{  
            while(index == 0){  
                System.out.println("!!!!!!!!!消费光了!!!!!!!!!");  
                this.wait();  
            }  
            this.notify();  
        }catch(InterruptedException e){  
            e.printStackTrace();  
        }catch(IllegalMonitorStateException e){  
            e.printStackTrace();  
        }  
        index--;  
        System.out.println("消费了:---------" + sm[index] + " 共" + index + "个馒头");  
        return sm[index];  
    }  
}  
 
class Producer implements Runnable  
{  
    StackBasket ss = new StackBasket();  
    Producer(StackBasket ss){  
        this.ss = ss;  
    }  
 
    /**   
    * show 生产进程.   
    */   
    public void run(){  
        for(int i = 0;i < 20;i++){  
            Mantou m = new Mantou(i);  
            ss.push(m);  
//          System.out.println("生产了:" + m + " 共" + ss.index + "个馒头");  
//          在上面一行进行测试是不妥的,对index的访问应该在原子操作里,因为可能在push之后此输出之前又消费了,会产生输出混乱  
            try{  
                Thread.sleep((int)(Math.random()*500));  
            }catch(InterruptedException e){  
                e.printStackTrace();  
            }  
        }  
    }  
}  
 
class Consumer implements Runnable  
{  
    StackBasket ss = new StackBasket();  
    Consumer(StackBasket ss){  
        this.ss = ss;  
    }  
 
    /**   
    * show 消费进程.  
    */   
    public void run(){  
        for(int i = 0;i < 20;i++){  
            Mantou m = ss.pop();  
//          System.out.println("消费了:---------" + m + " 共" + ss.index + "个馒头");  
//  同上  在上面一行进行测试也是不妥的,对index的访问应该在原子操作里,因为可能在pop之后此输出之前又生产了,会产生输出混乱  
            try{  
                Thread.sleep((int)(Math.random()*1000));  
            }catch(InterruptedException e){  
                e.printStackTrace();  
            }  
        }  
    }  
} 

原文链接: http://blog.csdn.net/shijinupc/article/details/7250407


转载地址:http://developer.51cto.com/art/201202/317209.htm

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值