java高并发(八)不可变对象
生活随笔
收集整理的這篇文章主要介紹了
java高并发(八)不可变对象
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
有一種對象一旦發(fā)布了,那么他就是安全對象,這就是不可變對象。
不可變對象需要滿足的條件:
- 對象創(chuàng)建以后其狀態(tài)就不能修改
- 對象所有的域都是final類型
- 對象是正確創(chuàng)建的(在對象創(chuàng)建期間,this引用沒有逸出)
final關鍵字:類、方法、變量
- 修飾類:不能被繼承。final類中的成員變量可以根據(jù)需要設置為final,要注意的是final類中的所有成員方法都會被隱式的指定為final方法。
- 修飾方法:1. 鎖定方法不被繼承類修改;2. 效率
- 修飾變量:基本數(shù)據(jù)類型變量在初始化之后就不能修改了,引用類型變量在初始化之后便不能指向另外一個對象
?下面舉例說明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引用變量不可以指定新的引用,但卻可以修改里面的值。
這樣就會引發(fā)線程安全方面的問題。
除了final定義不可變對象,是否還有其他手段定義不可變對象?當然可以
- Collections.unmodifiableXX: Collection、List、Set、Map......
- Guava:ImmutableXXX:Collection、List、Set、Map
?這樣運行就會報錯:
Exception in thread "main" java.lang.UnsupportedOperationExceptionat java.util.Collections$UnmodifiableMap.put(Collections.java:1457)at com.vincent.example.immutable.ImmutableExample2.main(ImmutableExample2.java:24)也就是說用Collections.unmodifiableMap聲明一個變量,他的內容就不可以修改了。數(shù)據(jù)不會被污染。
@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);} }根據(jù)變量實際情況變成最好變成不可變對象,如果可以盡量把對象變成不可變對象,這樣在多線程情況下就不會出現(xiàn)線程安全問題了。
與50位技術專家面對面20年技術見證,附贈技術全景圖總結
以上是生活随笔為你收集整理的java高并发(八)不可变对象的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: java高并发(七)发布对象
- 下一篇: java高并发(九)线程封闭