神奇的线程安全问题

    读完了《Java并发编程实践》这本书以后,以为对一般的线程安全问题有一个大概的理解,但是今天遇到的这个问题着实非常神奇,在书中也没有被提到过,这里特别记录下来。

 

public class Test {
    public void test1(O o) throws Exception {
        System.out.println("begin o:" + o.getId() + " " + o.getName());
        if (o.getId() == 1) {
            Thread.sleep(10000);
            System.out.println("o1:" + o.getId() + " " + o.getName());
        }

        if (o.getId() == 2) {
            System.out.println("o2:" + o.getId() + " " + o.getName());
        }
        System.out.println("end o:" + o.getId() + " " + o.getName());
    }

    public static void main(String[] args) throws Exception {

        final O o1 = new O(1, "o1");
        Thread t1 = new Thread() {
            @Override
            public void run() {
                try {
                    Test test = new Test();
                    test.test1(o1);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        };
        Thread t2 = new Thread() {
            @Override
            public void run() {
                try {
                    o1.setId(2);
                    o1.setName("o2");
                    Test test = new Test();
                    test.test1(o1);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        };
        t1.start();
        Thread.sleep(10);
        t2.start();

    }
}

class O {

    public O(int id, String name) {
        super();
        this.id = id;
        this.name = name;
    }

    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

 

    两个线程先后启动,传入的是不同状态的同一对象。在test1方法的第一条打印中可以看出来,参数对象的状态是不同的,但是当第一个线程在第一个判断处休眠第二个线程开始执行以后,第一个线程打印出来的状态就和之前不一致了,其状态和第二个参数对象的状态一样。按照理论,方法执行以后,JVM会在独立的线程中压入一个方法栈,栈中会包含一个方法执行所需要的所有元素(包括代码和参数),这里为什么会出现方法执行到一半的时候参数状态改变呢?

 

    推测:因为方法栈中压入的是一个对象的指针,而不是整个对象,所有当对象的状态被改变的时候还是可以第一时间被独立线程所察觉,所以尽管已经进入了方法栈,还是会引起对象状态的改变。

 

    以后在处理多线程的问题时,必须要所用无状态Immutable的VO对象。这样才能保证线程安全

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值