日期:2014-05-20  浏览次数:20753 次

多线程同步问题 高分
直接上代码:
两个类:
public class ThreadB extends Thread
{
  int total;
  public void run(){
  synchronized (this)
  {
  for(int i=0;i<101;i++){
  total+=i;
  }
  //notify();
  }
  }
  
}





public class ThreadA
{
   
  public static void main(String[] args)
  {
  ThreadB b = new ThreadB();
  b.start();
  // synchronized (b) //这里加上同步以后,下面的b.wait()注释掉,b.total输出0,如果加上输出5050 //{
  try{
  System.out.println("等待B");
  //b.wait();
  }catch (Exception e) {
  System.out.println(e);
  }
  System.out.println(b.total);
  //}
  }
}


反正这是看到的一个关于线程的wait(),notify()等待唤醒的例子,看完不是很理解,求解释,越详细越好

------解决方案--------------------
//这里加上同步以后,下面的b.wait()注释掉,b.total输出0,如果加上输出5050
因为这里是两个单独的线程,线程A打印输出的时候可能线程B还没有运行,所以打印出0。

而加上同步、wait方法之后,表示当前线程(线程A)等待线程B执行,直到有地方调用notify/notifyAll方法唤醒之前。线程A是不会执行的。这里没有地方调用notify/notifyAll方法,即是执行完线程B之后等待内置的通知。
这时total的值已经被改变了,所以打印出5050。
这里两个线程加上同步和wait()方法之后,其实就像一个单线程一样。


------解决方案--------------------
感觉这不是啥好例子.其实你说:这里加上同步以后,下面的b.wait()注释掉,b.total输出0
这只是因为基本上你看到的都是main这个线程先被执行了,执行完打印total的语句然后b启动的线程才有机会得到CUP执行,如果你用b.start()启动一个线程后,用Thread.sleep(100);让当前线程睡眠一下,b启动的线程已经算完了,然后打印的就是5050了

至于你说假设wait()后打印出5050,同样也是main线程先得到机会执行了,获得b的监视锁,然后它用wait释放说,而b启动线程刚好抢到锁,然后得到调度执行,计算total,最后打印出5050

反正看着这代码挺不爽的,加入我让main线程睡一下,让b启动的线程先执行,然后main线程再执行,调用wait进入等待,但b已经执行完,所以就再也没有别人去唤醒它了,永远停在那,如下:
Java code

class ThreadB extends Thread {
    int total;

    public void run() {
        /*try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }*/
        synchronized (this) {
            for (int i = 0; i < 101; i++) {
                total += i;
            }
             notify();
        }
    }

}

public class ThreadA {

    public static void main(String[] args) throws Exception {
        ThreadB b = new ThreadB();
        b.start();
        Thread.sleep(100);
        synchronized (b) {// 这里加上同步以后,下面的b.wait()注释掉,b.total输出0,如果加上输出5050 //{
            try {
                System.out.println("等待B");
                b.wait();
                System.out.println("shdhihhi");
            } catch (Exception e) {
                System.out.println(e);
            }
            System.out.println(b.total);
        }
    }
}