多线程中的生产者与消费者的问题,重点是统一资源的使用,涉及线程同步。课堂练习习题之牛的培养和宰杀来代替生产和消费。
首先定义的是一个牛(ox)类,包括牛的年龄和体重;其次,是牛的培养工厂(oxPen),在工厂中,分别存储成牛和牛犊,使用集合,因为牛犊和成牛的数量是动态变化,牛犊的初始化变量是100只。然后是培养方法和宰杀方法
培养方法:当牛犊的集合大于0时代表有牛犊可以培养,此时,就移除一只牛犊,然后将体重赋值200,年龄超过4,代表牛犊成了成牛,然后将此实例放入成牛的集合。
代码如下:`public synchronized void growOx() {
if (oxList.size() > 0) {
Ox temp = oxList.remove(0);
temp.setAge(4);
temp.setWeight(200);
oxvector.add(temp);
System.out.println("生产----牛栏剩余牛犊:" + oxList.size());
System.out.println("生产----成牛数量:" + oxvector.size());
}
}`
宰杀方法:也就是我们的消费方法,取出成牛集合中的牛,然后判断是否体重大于200,年龄大于4,为真则将牛杀了,移除该元素。
public synchronized void killOx() {
Ox tempOx = oxvector.get(0);
if (tempOx != null && tempOx.getAge() >= 3 && tempOx.getWeight() >= 200) {
oxvector.remove(0);
System.out.println("消费----成牛剩余数量:" + oxvector.size());
}
}
这里的宰杀和培养方法都是锁着的,防止出现生产或者宰杀时,牛的数量为负,增加了线程安全。
然后我们写了两个线程一个是消费线程,一个是生产线程:
分别实现牛的生产和消费;
当牛犊的集合长度大于0的时候,方能进行成牛的培养;
而成牛的消费则是成牛的集合的数量大于0的时候才可以进行消费,当成牛的数量小于0并且牛犊的集合中还有牛时,进行线程等待,
public class ConsumerRunable implements Runnable{
OxPen oxPen;
public ConsumerRunable(OxPen oxPen) {
this.oxPen=oxPen;
}
@Override
public void run() {
//循环条件是成牛的数量大于0或者牛犊的数量大于0
//当成牛的数量小于0时,同时牛犊的数量大于0时,进行线程睡眠,进行牛的成长
while(oxPen.oxlist.size()>0||oxPen.oxVector.size()>0){
if(oxPen.oxVector.size()<=0&&oxPen.oxlist.size()>0){
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}else if(oxPen.oxlist.size()>0){
oxPen.killOx();
}
}
}
}
生产线程:
package main03.review02;
//消费进程
public class ProduceRunable implements Runnable{
OxPen oxPen;
public ProduceRunable(OxPen oxPen) {
this.oxPen=oxPen;
}
@Override
public void run() {
while(oxPen.oxlist.size()>0){
try {
Thread.sleep(300);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
oxPen.growOx();
}
}
}
博主出现了一个问题就是没有设置消费者线程的中的判断;如果不判断成牛的数量直接使用的话,在宰杀方法中,会出现娶不到第一个元素,从而报错。
while(oxPen.oxlist.size()>0||oxPen.oxVector.size()>0){
if(oxPen.oxVector.size()<=0&&oxPen.oxlist.size()>0){
就是这几行元素。