synchronized 使用可变Integer静态变量作为锁的问题
package com.i3geek.pattern.singleton.test;

public class MyTest {

    public static void main(String[] args) {
        new MyThread("kang-01").start();
        new MyThread("kang-02").start();
        new MyThread("kang-03").start();
    }

}
class MyThread extends Thread{

    private static Integer i = 1;
    private static Object obj = new Object();

    public MyThread() {
    }
    public MyThread(String threadName) {
        super.setName(threadName);
    }

    @Override
    public void run() {
        show();
    }

    public void show(){
        for(boolean flag = true; flag;){
            synchronized (obj) {  //问题:这里的使用obj作为锁是线程安全的?使用i作为线程是不安全的?
                if(i<=50){
                    System.out.println(Thread.currentThread().getName()+" - " + i++);
                    try {
                        Thread.currentThread().sleep(500);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }else{
                    flag = false;
                }
            }
        }
    }
}

经过测试发现使用 i 变量作为锁不是线程安全的,而使用obj确是线程安全的

1个回答

我觉得这个问题有点搞笑, Integer 继承自 Object(Object都可以,Integer怎么会不行?), 并且Integer i = 1, 会使用IntegerCache, 将i 指向同一个 1 的包装类型常量, 怎么会不行呢?

问题在于: 你在循环内部,改变了 i 的值,不断的改变锁,当然就出问题了,锁是不能改变的!

qq_25272021
qq_25272021 回复hot2dog: 嗯嗯,试了一下,是可以的,我明白了,谢谢你解答^_^
一年多之前 回复
qq_27848369
hot2dog 回复qq_25272021: 这种是可以的, 锁的引用没变
一年多之前 回复
qq_25272021
qq_25272021 那我新建一个对象 加入private static Person person =new Person(name:"xiaoming"),以这个为锁对象,synchronized (person) {...; person.setName(i+"") //此处改变name的值,那应用是线程安全吗? ...;}
一年多之前 回复
Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
立即提问
相关内容推荐