实现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);
}
}
}