经典线程处理打印ABC问题分析

来源:互联网 发布:英菲克爬山虎软件下载 编辑:程序博客网 时间:2024/05/19 23:15

package com;

public class MyThreadPrinter2 implements Runnable {

 private String name;
 private Object prev;
 private Object self;

 private MyThreadPrinter2() {

 }

 private MyThreadPrinter2(String name, Object prev, Object self) {
  this.name = name;
  this.prev = prev;
  this.self = self;
 }

 @Override
 public void run() {
  int count = 10;
  while (count > 0) {
   synchronized (prev) {
    synchronized (self) {
     System.out.print(name);
     count--;

     self.notify();
    }
    try {
     prev.wait();
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
   }

  }
 }

 public static void main(String[] args) throws Exception {
  Object a = new Object();
  Object b = new Object();
  Object c = new Object();
  MyThreadPrinter2 pa = new MyThreadPrinter2("A", c, a);
  MyThreadPrinter2 pb = new MyThreadPrinter2("B", a, b);
  MyThreadPrinter2 pc = new MyThreadPrinter2("C", b, c);

  new Thread(pa).start();
  new Thread(pb).start();
  new Thread(pc).start();
 }
}
/*
 *
 *
 * 思路分析:
 * 该问题为三个线程间的同步唤醒操作,主要的目的就是ThreadA->TthreadB->ThreadC->ThreadA循环执行单个线程.
 * 为了控制线程执行的顺序
 * 就必须确定唤醒,等待的顺序,所以每一个线程必须同时持有两个对象锁,才能继续执行,
 * 一个对象锁是prev,就是前一个线程所持有的对象锁.还有一个就是自身对象锁,
 * 主要的思想就是,为了控制执行的顺序,必须先持有prev锁,也就是前一个线程要释放自身
 * 的对象锁,再去申请自身的对象锁,两者兼备是打印,之后首先调用self.notify()释放自身对象锁,
 * 唤醒下一个线程等待,在调用prev.wait()释放prev对象锁,终止当前进程,等待循环结束再次被唤醒
 *
 *
 *
*/

0 0
原创粉丝点击