java多线程 生产者消费者_java多线程之生产者消费者经典问题 - 很不错的范例

/**生产者消费者问题,涉及到几个类

* 第一,这个问题本身就是一个类,即主类

* 第二,既然是生产者、消费者,那么生产者类和消费者类就是必须的

* 第三,生产什么,消费什么,所以物品类是必须的,这里是馒头类

* 第四,既然是线程,那么就不是一对一的,也就是说不是生产一个消费一个,既然这样,多生产的往哪里放,

*      现实中就是筐了,在计算机中也就是数据结构,筐在数据结构中最形象的就是栈了,因此还要一个栈类

*/

packagethread;

publicclassProduceConsume {

publicstaticvoidmain(String[] args) {

SyncStack ss = newSyncStack();//建造一个装馒头的框

Producer p = newProducer(ss);//新建一个生产者,使之持有框

Consume c = newConsume(ss);//新建一个消费者,使之持有同一个框

Thread tp = newThread(p);//新建一个生产者线程

Thread tc = newThread(c);//新建一个消费者线程

tp.start();//启动生产者线程

tc.start();//启动消费者线程

}

}

//馒头类

classSteamBread{

intid;//馒头编号

SteamBread(intid){

this.id = id;

}

publicString toString(){

return"steamBread:"+id;

}

}

//装馒头的框,栈结构

classSyncStack{

intindex =0;

SteamBread[] stb = newSteamBread[6];//构造馒头数组,相当于馒头筐,容量是6

//放入框中,相当于入栈

publicsynchronizedvoidpush(SteamBread sb){

while(index==stb.length){//筐满了,即栈满,

try{

this.wait();//让当前线程等待

} catch(InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

this.notify();//唤醒在此对象监视器上等待的单个线程,即消费者线程

stb[index] = sb;

this.index++;

}

//从框中拿出,相当于出栈

publicsynchronizedSteamBread pop(){

while(index==0){//筐空了,即栈空

try{

this.wait();

} catch(InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

this.notify();

this.index--;//push第n个之后,this.index++,使栈顶为n+1,故return之前要减一

returnstb[index];

}

}

//生产者类,实现了Runnable接口,以便于构造生产者线程

classProducerimplementsRunnable{

SyncStack ss = null;

Producer(SyncStack ss){

this.ss = ss;

}

@Override

publicvoidrun() {

// 开始生产馒头

for(inti=0;i<20;i++){

SteamBread stb = newSteamBread(i);

ss.push(stb);

System.out.println("生产了"+stb);

try{

Thread.sleep(10);//每生产一个馒头,睡觉10毫秒

} catch(InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

}

//消费者类,实现了Runnable接口,以便于构造消费者线程

classConsumeimplementsRunnable{

SyncStack ss = null;

publicConsume(SyncStack ss) {

super();

this.ss = ss;

}

@Override

publicvoidrun() {

// TODO Auto-generated method stub

for(inti=0;i<20;i++){//开始消费馒头

SteamBread stb = ss.pop();

System.out.println("消费了"+stb);

try{

Thread.sleep(100);//每消费一个馒头,睡觉100毫秒。即生产多个,消费一个

} catch(InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

}

运行结果:

1352376576_9997.png

1352376582_5734.png

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值