class SyncStack{ //同步堆栈类 private int index = 0; //堆栈指针初始值为0 private char []buffer = new char[6]; //堆栈有6个字符的空间 public synchronized void push(char c){ //加上互斥锁 while(index = = buffer.length){ //堆栈已满,不能压栈 try{ this.wait(); //等待,直到有数据出栈 }catch(InterruptedException e){} } this.notify(); //通知其它线程把数据出栈 buffer[index] = c; //数据入栈 index++; //指针向上移动 } public synchronized char pop(){ //加上互斥锁 while(index ==0){ //堆栈无数据,不能出栈 try{ this.wait(); //等待其它线程把数据入栈 }catch(InterruptedException e){} } this.notify(); //通知其它线程入栈 index- -; //指针向下移动 return buffer[index]; //数据出栈 } } class Producer implements Runnable{ //生产者类 SyncStack theStack; //生产者类生成的字母都保存到同步堆栈中 public Producer(SyncStack s){ theStack = s; } public void run(){ char c; for(int i=0; i<20; i++){ c =(char)(Math.random()*26+'A'); //随机产生20个字符 theStack.push(c); //把字符入栈 System.out.println("Produced: "+c); //打印字符 try{ Thread.sleep((int)(Math.random()*1000)); /*每产生一个字符线程就睡眠*/ }catch(InterruptedException e){} } } } class Consumer implements Runnable{ //消费者类 SyncStack theStack; //消费者类获得的字符都来自同步堆栈 public Consumer(SyncStack s){ theStack = s; } public void run(){ char c; for(int i=0;i<20;i++){ c = theStack.pop(); //从堆栈中读取字符 System.out.println("Consumed: "+c); //打印字符 try{ Thread.sleep((int)(Math.random()*1000)); /*每读取一个字符线程就睡眠*/ }catch(InterruptedException e){} } } } public class SyncTest{ public static void main(String args[]){ SyncStack stack = new SyncStack(); //下面的消费者类对象和生产者类对象所操作的是同一个同步堆栈对象 Runnable source=new Producer(stack); Runnable sink = new Consumer(stack); Thread t1 = new Thread(source); //线程实例化 Thread t2 = new Thread(sink); //线程实例化 t1.start(); //线程启动 t2.start(); //线程启动 } }
类Producer是生产者模型,其中的 run()方法中定义了生产者线程所做的操作,循环调用push()方法,将生产的20个字母送入堆栈中,每次执行完push操作后,调用sleep()方法睡眠一段随机时间,以给其他线程执行的机会。类Consumer是消费者模型,循环调用pop()方法,从堆栈中取出一个数据,一共取20次,每次执行完pop操作后,调用sleep()方法睡眠一段随机时间,以给其他线程执行的机会。
程序执行结果
Produced:V
Consumed:V
Produced:E
Consumed:E
Produced:P
Produced:L
...
Consumed:L
Consumed:P
在上述的例子中,通过运用wait()和notify()方法来实现线程的同步,在同步中还会用到notifyAll()方法,一般来说,每个共享对象的互斥锁存在两个队列,一个是锁等待队列,另一个是锁申请队列,锁申请队列中的第一个线程可以对该共享对象进行操作,而锁等待队列中的线程在某些情况下将移入到锁申请队列。下面比较一下wait()、notify()和notifyAll()方法:
(1) wait,nofity,notifyAll必须在已经持有锁的情况下执行,所以它们只能出现在synchronized作用的范围内,也就是出现在用 synchronized修饰的方法或类中。
(2) wait的作用:释放已持有的锁,进入等待队列.
(3) notify的作用:唤醒wait队列中的第一个线程并把它移入锁申请队列.
(4) notifyAll的作用:唤醒wait队列中的所有的线程并把它们移入锁申请队列.