当前位置: 代码迷 >> J2SE >> 被synchronized块修饰的类成员变量和声明为static的类成员变量一样?该如何处理
  详细解决方案

被synchronized块修饰的类成员变量和声明为static的类成员变量一样?该如何处理

热度:24   发布时间:2016-04-24 01:03:09.0
被synchronized块修饰的类成员变量和声明为static的类成员变量一样?
实例如下,定义了一个线程类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),所以需要加同步块。
------解决方案--------------------
探讨
实例如下,定义了一个线程类MyThread,含有一个成员变量i,没有声明为static,为什么在synchronized块内操作后,类对象B得到的i值是上一个类对象A修改过的值呢?
这样不就和static类成员一样了吗?synchronized有此作用?

Java code
public class JavaTest {

/**
* @param args
……
  相关解决方案