java高併發(八)不可變物件
阿新 • • 發佈:2019-08-07
有一種物件一旦釋出了,那麼他就是安全物件,這就是不可變物件。
不可變物件需要滿足的條件:
- 物件建立以後其狀態就不能修改
- 物件所有的域都是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);
}
}
根據變數實際情況變成最好變成不可變物件,如果可以儘量把物件變成不可變物件,這樣在多執行緒情況下就不會出現執行