1. 程式人生 > >java高併發(八)不可變物件

java高併發(八)不可變物件

有一種物件一旦釋出了,那麼他就是安全物件,這就是不可變物件。

不可變物件需要滿足的條件:

  • 物件建立以後其狀態就不能修改
  • 物件所有的域都是final型別
  • 物件是正確建立的(在物件建立期間,this引用沒有逸出)

final關鍵字:類、方法、變數

  • 修飾類:不能被繼承。final類中的成員變數可以根據需要設定為final,要注意的是final類中的所有成員方法都會被隱式的指定為final方法。
  • 修飾方法:1. 鎖定方法不被繼承類修改;2. 效率
  • 修飾變數:基本資料型別變數在初始化之後就不能修改了,引用型別變數在初始化之後便不能指向另外一個物件

 下面舉例說明final修飾變數:

@Slf4j
@NotThreadSafe
public class ImmutableExample1 {

    private final static Integer a = 1;
    private final static String b = "2";
    private final static Map<Integer, Integer> map = new HashMap<>();

    static {
        map.put(1, 2);
        map.put(2, 3);
    }

    public static void main(String[] args) {
//        a = 2;
//        b = "3";
//        map  = new HashMap<>();
        map.put(1, 3);
        log.info("{}", map.get(1));
    }

}

map引用變數不可以指定新的引用,但卻可以修改裡面的值。

這樣就會引發執行緒安全方面的問題。

除了final定義不可變物件,是否還有其他手段定義不可變物件?當然可以

  • Collections.unmodifiableXX: Collection、List、Set、Map......
  • Guava:ImmutableXXX:Collection、List、Set、Map
@Slf4j
@ThreadSafe
public class ImmutableExample2 {

    private static Map<Integer, Integer> map = new HashMap<>();

    static {
        map.put(1, 2);
        map.put(2, 3);
        map = Collections.unmodifiableMap(map);
    }

    public static void main(String[] args) {
        map.put(1, 3);
        map.put(3,4);
        log.info("{}", map.get(3));
    }
}

 這樣執行就會報錯:

Exception in thread "main" java.lang.UnsupportedOperationException
	at java.util.Collections$UnmodifiableMap.put(Collections.java:1457)
	at com.vincent.example.immutable.ImmutableExample2.main(ImmutableExample2.java:24)

也就是說用Collections.unmodifiableMap宣告一個變數,他的內容就不可以修改了。資料不會被汙染。

@ThreadSafe
public class ImmutableExample3 {

    private final static ImmutableList<Integer> list = ImmutableList.of(1,2,3);
    private final static ImmutableSet<Integer> set = ImmutableSet.copyOf(list);
    private final static ImmutableMap<Integer, Integer> map = ImmutableMap.of(1,2,3,4);
    private final static ImmutableMap<Integer, Integer> map2 = ImmutableMap.<Integer, Integer>builder().put(1,2).put(3,4).build();

    public static void main(String[] args) {
        map2.put(4,5);
    }
}

根據變數實際情況變成最好變成不可變物件,如果可以儘量把物件變成不可變物件,這樣在多執行緒情況下就不會出現執行