实例如下,定义了一个线程类MyThread,含有一个成员变量i,没有声明为static,为什么在synchronized块内操作后,类对象B得到的i值是上一个类对象A修改过的值呢?
这样不就和static类成员一样了吗?synchronized有此作用?
- Java code
public class JavaTest { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub System.out.println("main start"); MyThread thread = new MyThread(); Thread a = new Thread(thread, "A"); Thread b = new Thread(thread, "B"); a.start(); b.start(); System.out.println("main end"); }}class MyThread implements Runnable { private int i = 0; @Override public void run() { // TODO Auto-generated method stub System.out.println(Thread.currentThread().getName() + " run enter i = " + i); synchronized(this){ System.out.println(Thread.currentThread().getName() + " synchronized >>>>>>> i = " + i); for (; i < 5; i++) { //for (i = 0; i < 5; i++) { System.out.println(Thread.currentThread().getName() + " synchronized loop " + i); } } System.out.println(Thread.currentThread().getName() + " run exit"); }}
输出结果:
main start
main end
A run enter i = 0
B run enter i = 0
A synchronized >>>>>>> i = 0
A synchronized loop 0
A synchronized loop 1
A synchronized loop 2
A synchronized loop 3
A synchronized loop 4
A run exit
B synchronized >>>>>>> i = 5 // 为什么i的值是5而不是0呢? 这样不就和static成员变量一样了?
B run exit
------解决方案--------------------
虽然有2个线程 , 但操作的是同一个对象。
------解决方案--------------------
就相当于两个线程在轮番使用这个对象,只有一个对象
------解决方案--------------------
------解决方案--------------------
Thread a = new Thread(thread, "A"); //a线程调用的是thread对象的run方法
Thread b = new Thread(thread, "B"); //b线程调用的也是thread对象的run方法
两个线程使用的都是同一个thread对象,所以两个线程之间相互产生影响
------解决方案--------------------
有两点:
第一,线程a和线程b同时使用thread对象,当线程a退出同步块后,i的值变成5
第二,由于同步块对变量的i的改变对其他线程可见,效果比volatile稍强,故其他线程也能看到这个值已经发生改变的值。
因为两个线程同时操作一个对象的同一个状态(这里是i),所以需要加同步块。
------解决方案--------------------