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

实现Runnable接口的线程类,很困惑的问题
public class SecondThread implements Runnable{  
  int i;  
@Override
public void run() {
for(;i<200;i++){
System.out.println(Thread.currentThread().getName()+" "+i);

}

}

public static void main(String[] args) {
for(int i=0;i<30;i++){
//System.out.println(Thread.currentThread().getName()+" "+i);

if(i==20){
SecondThread st=new SecondThread();
new Thread(st,"新线程1").start();
new Thread(st,"新线程2").start();

}

}

}


}

说实现了Runnable接口方式创建的多个线程可以共享线程类的实例属性,上面的代码,我运行了好几遍发现,线程1和线程2都有可能出现相同的i值,既然是共享实例属性,那应该是2个线程共享st的i值,既然都是用的同一个对象的i值,为什么还出现相同的i值呢?那不是与“实现了Runnable接口方式创建的多个线程可以共享线程类的实例属性”这句话矛盾了?求师傅帮忙解惑一下,我晕了一下午了这个点

------解决方案--------------------
并发就是这样,你没有做并发控制,就会有各种灵异现象,因为线程执行是交错执行的。

也就意味着,两个线程可能分别执行了i++后,然后才执行了 println,给个典型交错场景
T1:i++ // i为1
T2:i++ // i为2
T1:System.out.println(Thread.currentThread().getName()+" "+i); // 输出2
T2:System.out.println(Thread.currentThread().getName()+" "+i); // 输出2

要么你就严格的做并发控制,比如:

Java code
public void run() {
  synchronized(this) {
    for(;i<200;i++){
      System.out.println(Thread.currentThread().getName()+" "+i);
    }
  }
}