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接口方式创建的多个线程可以共享线程类的实例属性”这句话矛盾了?求师傅帮忙解惑一下,我晕了一下午了这个点

解决方案 »

  1.   

    并发就是这样,你没有做并发控制,就会有各种灵异现象,因为线程执行是交错执行的。也就意味着,两个线程可能分别执行了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要么你就严格的做并发控制,比如:public void run() {
      synchronized(this) {
        for(;i<200;i++){
          System.out.println(Thread.currentThread().getName()+" "+i);
        }
      }
    }
      

  2.   

    在方法上面或者块上加synchronized关键字后,防止线程同步。
      

  3.   

    请教一下,就是后面的for循环是不是没有用啊,只有run()方法结束了,程序也就结束了???不会说是i没增加1就执行一次线程??