這兩天覆習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 }