基於java callable及future接口解決生產者消費者問題

這兩天覆習java線程時,把java裏面的線程基本知識點與jdk1.5之後新添加的一些類的使用都瞭解了一下,借用生產者消費者的問題來將他們實踐一下。java

 

題目:(題目在csdn一大牛的空間找的)dom

     生產者-消費者模式,這個食堂中只有1張桌子,同時最多放10個盤子,如今有4個廚師作菜,每作好一盤就往桌子上放(生產者將產品往倉庫中放),而有6個食客不停地吃(消費者消費產品,爲了說明問題,他們的食量是無的).通常而言,廚師200-400ms作出一盤菜,而食客要400-600ms吃完一盤.當桌子上放滿了10個盤子後,全部廚師都不能再往桌子上放,而當桌子上沒有盤子時,全部的食客都只好等待.ide

代碼以下:this

 

  1 package net.xicp.tarbitrary.producerandcustomer;
  2 
  3 import java.util.LinkedList;
  4 import java.util.Random;
  5 import java.util.concurrent.Callable;
  6 import java.util.concurrent.ExecutionException;
  7 import java.util.concurrent.ExecutorService;
  8 import java.util.concurrent.Executors;
  9 import java.util.concurrent.FutureTask;
 10 import java.util.concurrent.TimeUnit;
 11 import java.util.concurrent.locks.Condition;
 12 import java.util.concurrent.locks.Lock;
 13 import java.util.concurrent.locks.ReentrantLock;
 14 
 15 
 16 public class NewFeatureTest {
 17 
 18  public static void main(String[] args) throws InterruptedException,
 19    ExecutionException {
 20   // 建立一個具備10個線程的線程池
 21   ExecutorService se = Executors.newFixedThreadPool(30);
 22   // 產生一張桌子,桌子的最大容量爲10
 23   DiningTable table = new DiningTable(10);
 24   // 產生生產者對象
 25   Chef chef = new Chef(table);
 26   // 產生消費者對象
 27   Eater eater = new Eater(table);
 28   
 29   
 30   //本例只是模擬10個生產者與10個消費者,並無徹底按照題目的意思來進行解答
 31   for (int i = 1; i < 11; i++) {
 32    final FutureTask cheafTask = new FutureTask(chef);
 33    final FutureTask eaterTask = new FutureTask(eater);
 34    // 啓動一個生產者線程
 35    se.execute(cheafTask);
 36    // 啓動一個消費者線程
 37    se.execute(eaterTask);
 38    // 啓動一個線程監聽當前生產者線程的結果
 39    se.execute(new Runnable() {
 40     @Override
 41     public void run() {
 42      try {
 43       String st = cheafTask.get();
 44       System.out.println(st);
 45      } catch (InterruptedException e) {
 46       e.printStackTrace();
 47      } catch (ExecutionException e) {
 48       e.printStackTrace();
 49      }
 50 
 51     }
 52    });
 53    // 啓動一個線程監聽當前消費者線程的結果
 54    se.execute(new Runnable() {
 55     @Override
 56     public void run() {
 57      try {
 58       String st = eaterTask.get();
 59       System.out.println(st);
 60      } catch (InterruptedException e) {
 61       e.printStackTrace();
 62      } catch (ExecutionException e) {
 63       e.printStackTrace();
 64      }
 65     }
 66    });
 67    //有序地關閉前面已經啓動的一些線程
 68   }
 69   se.shutdown();
 70  }
 71 }
 72 
 73 
 74 class Chef implements Callable {
 75  private DiningTable t;
 76  private Random random = new Random();
 77 
 78  public Chef(DiningTable table) {
 79   this.t = table;
 80  }
 81 
 82  public DiningTable getT() {
 83   return t;
 84  }
 85 
 86  public void setT(DiningTable t) {
 87   this.t = t;
 88  }
 89 
 90  public Dish produce() {
 91   try {
 92    // 模擬作菜用的時間,200-400毫秒一道菜
 93    // 跟下面一句的效果一致
 94    TimeUnit.MILLISECONDS.sleep(200 + random.nextInt(201));
 95    // Thread.sleep(200 + random.nextInt(201));
 96   } catch (InterruptedException e) {
 97    e.printStackTrace();
 98   }
 99   Dish food = new Dish();
100   return food;
101  }
102 
103  @Override
104  public String call() throws Exception {
105   for (int i = 0; i < 10; i++) {
106    Dish food = produce();
107    t.putFood(food);
108   }
109   return Thread.currentThread().getName() + "結束";
110  }
111 
112 }
113 
114 
115 class Eater implements Callable {
116  private DiningTable t;
117  private Random random = new Random();
118 
119  public Eater(DiningTable table) {
120   this.t = table;
121  }
122 
123  public DiningTable getT() {
124   return t;
125  }
126 
127  public void setT(DiningTable t) {
128   this.t = t;
129  }
130 
131  
132  public void consume(Dish food) {
133   try {
134    // 模擬吃菜用的時間
135    // 跟下面一句的效果一致
136    TimeUnit.MILLISECONDS.sleep(400 + random.nextInt(201));
137    // Thread.sleep(400 + random.nextInt(201));
138   } catch (InterruptedException e) {
139    e.printStackTrace();
140   }
141  }
142 
143  @Override
144  public String call() throws Exception {
145   for (int i = 0; i < 10; i++) {
146    Dish food = t.getFood();
147    consume(food);
148   }
149 
150   return Thread.currentThread().getName() + "結束";
151  }
152 
153 }
154 
155 
156 class Dish {
157  Random random = new Random();
158 
159  private String foodName;
160 
161  public Dish() {
162   super();
163   int posi = random.nextInt(name.length);
164   this.foodName = name[posi];
165  }
166 
167  String name[] = { "巫山烤魚", "清炒小白菜 ", "糖醋里脊", "手撕包菜", "拔絲香蕉", "水煮肉片", "酸菜魚",
168    "土豆肉片" };
169 
170  public String getName() {
171   return foodName;
172  }
173 
174 }
175 
176 
177 class DiningTable {
178  // 裝食物用的容器
179  private LinkedList list = new LinkedList();
180  // 定義最大放菜量
181  private int maxSize;
182  // 建立一把鎖
183  private Lock lock = new ReentrantLock();
184  // 得到當前lock對象的 Condition 實例
185  private Condition condition;
186 
187  public Lock getLock() {
188   return lock;
189  }
190 
191  public void setLock(Lock lock) {
192   this.lock = lock;
193  }
194 
195  public Condition getCondition() {
196   return condition;
197  }
198 
199  public void setCondition(Condition condition) {
200   this.condition = condition;
201  }
202 
203  public LinkedList getList() {
204   return list;
205  }
206 
207  public void setList(LinkedList list) {
208   this.list = list;
209  }
210 
211  public int getMaxSize() {
212   return maxSize;
213  }
214 
215  
216  public void setMaxSize(int maxSize) {
217   this.maxSize = maxSize;
218  }
219 
220  public DiningTable(int maxSize) {
221   super();
222   this.maxSize = maxSize;
223   this.condition = lock.newCondition();
224  }
225 
226  
227  public void putFood(Dish food) {
228   // 先進行鎖定, 與synchronize起的做用相似
229   lock.lock();
230   try {
231    // 防止假喚醒
232    while (list.size() >= maxSize) {
233 
234     System.out.println( "菜已經上滿了" + Thread.currentThread().getName() + ",讓他們先吃,等消滅幾盤再作菜");
235     // 若是桌子的菜已經放滿了,則讓當前線程等待
236     this.condition.await();
237    }
238    list.addLast(food);
239    System.out.println(Thread.currentThread().getName() + "烹飪了"
240      + food.getName());
241    // 通知全部休眠的食客線程繼續吃菜
242    this.condition.signalAll();
243   } catch (Exception e) {
244    e.printStackTrace();
245   } finally {
246    // 解除鎖定
247    lock.unlock();
248   }
249  }
250 
251  
252  public Dish getFood() {
253   // 先進行鎖定, 與synchronize起的做用相似
254   lock.lock();
255   Dish dish = null;
256   try {
257    // 防止假喚醒
258    while (list.size() <= 0) {
259     System.out.println( "菜已經被吃光了" + Thread.currentThread().getName() + "歇會再吃");
260 
261      // 若是桌子上已經沒有菜了,則讓當前線程等待
262      this.condition.await();
263     }
264    dish = list.removeFirst();
265    System.out.println(Thread.currentThread().getName() + "吃掉了"
266      + dish.getName());
267    // 通知全部休眠的作菜線程繼續作菜
268    this.condition.signalAll();
269   } catch (Exception e) {
270    e.printStackTrace();
271   } finally {
272    // 解除鎖定
273    lock.unlock();
274   }
275 
276   return dish;
277  }
278 
279 }
相關文章
相關標籤/搜索