搜索

首页  >  问答  >  正文

java - String是线程安全的吗?那String岂不是不能进行同步?

  1. 我们知道不变对象是指一旦创建不能修改内部状态的对象,因为不变对象没有提供可供修改内部状态的方法,所以不变对象是线程安全的。但是String,当然包括其他的基本数据的包装类,如Integer/Long/Float等等,也是不变对象,但却是可以修改值的,这怎么保证线程安全?

public class Test{

private String str;

public void test(){
        if(("").equals(str)){  //多个线程同时判断,可能导致多次执行
            str = "1";
            //do something
        }
}

}

怪我咯怪我咯2766 天前614

全部回复(2)我来回复

  • 天蓬老师

    天蓬老师2017-04-18 10:29:08

    重新找个可以锁住的对象,锁住它再修改String。

    private String lock = new Object();
    private String str;
    public void test(){
        synchronized(lock) {
            if(("").equals(str)){  //多个线程同时判断,可能导致多次执行
                str = "1";
                //do something
            }
        }
    }

    回复
    0
  • 阿神

    阿神2017-04-18 10:29:08

    Java中,String类型对象本身是不可变的。因为String会被存储到一个叫做常量池的内存区域。
    比如

            String a="test";
            String b="test";
            System.out.println(a==b);
            System.out.println(System.identityHashCode(a));
            System.out.println(System.identityHashCode(b));

    输出:

    true
    851664923
    851664923

    那么为什么你又感觉可变?

    String a="test";
    a="test1"
    

    然后a就变成了test1,其实在这里是新建了一个"test1"字符串对象(如果常量池没有这个值的话就是新建)。然后将变量引用指向它。注意:这里并没有修改"test"这个变量的内部状态,"test"这个字符串对象是线程安全的。
    除非你用final修饰,否则所有的变量指向都是可变的。
    这种情况下要保证线程安全性:
    第0:可以考虑使用volatile确保可见性。
    第一可以使用final修饰
    第二你可以使用AtomicReference之类的原子对象,对于Integer等也有AtomicInteger之类的
    第三对相应代码区域加锁

    回复
    0
  • 取消回复