1 重点:
1.1 3 不可变集合应用场景
1.2 7 demo之 JDK提供的方法实现不可变集合 demo
1.3 8 demo之 guava提供方法实现不可变集合 demo
1.4 8 demo之 guava提供方法实现不可变集合 guava依赖
2 不可变集合简介
创建对象的不可变拷贝是一项很好的防御性编程技巧。
Guava为所有JDK标准集合类型和Guava新集合类型都提供了简单易用的不可变版本。
3 不可变集合应用场景
需要根据具体的业务场景灵活使用。比如在查询用户订单列表的时候,我们希望返回的订单列表在后续的程序处理时不能对它进行增减
(把不是你的订单放进去或者将你的某些订单删除掉),就需要使用不可变集合来限制下游程序操作。
比如放款通知书,不可变更。
4 不可变集合优点
◆当对象被不可信的库调用时,不可变形式是安全的
◆不可变对象被多个线程调用时,不存在竞态条件问题
◆不可变集合不需要考虑变化,因此可以节省时间和空间。
◆不可变对象因为有固定不变,可以作为常量来安全使用。
5 JDK提供的unmodifiableXXX方法(该方法也可以实现不可变集合) 缺点
◆笨重而且累赘
◆不安全
◆低效
6 demo之 可变集合(无法实现不可变需求)
/** * 1 可变集合demo */ @Test public void testCanChangeCollect(){ List<Integer> list1 = new ArrayList<Integer>(); list1.add(1); list1.add(2); list1.add(3); testCanChange(list1); System.out.println(JSONObject.toJSONString(list1)); } public void testCanChange(List<Integer> list1){ list1.remove(0); }
打印日志:
[2,3]
Process finished with exit code 0
7 demo之 JDK提供的方法实现不可变集合
/** * 2 jdk原生改为不可变集合 */ @Test public void jdkOriginalNoChangeCollectTest(){ List<Integer> list1 = new ArrayList<Integer>(); list1.add(1); list1.add(2); list1.add(3); List<Integer> list2 = Collections.unmodifiableList(list1); testCanChange(list2); System.out.println(JSONObject.toJSONString(list1)); } public void testCanChange(List<Integer> list1){ list1.remove(0); }
打印日志:
java.lang.UnsupportedOperationException at java.base/java.util.Collections$UnmodifiableList.remove(Collections.java:1318) at com.imooc.zhangxiaoxi.guava.ImmutableTest.testCanChange(ImmutableTest.java:38) at com.imooc.zhangxiaoxi.guava.ImmutableTest.jdkOriginalNoChangeCollectTest(ImmutableTest.java:52) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:564) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) at org.junit.runners.ParentRunner.run(ParentRunner.java:363) at org.junit.runner.JUnitCore.run(JUnitCore.java:137) at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68) at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47) at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242) at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70) Process finished with exit code -1
8 demo之 guava提供方法实现不可变集合
/** * 3 guava 不可变集合的应用,guava 创建不可变集合的三种方式 */ @Test public void testGuavaNoChangeCollect(){ List<Integer> list1 = new ArrayList<Integer>(); list1.add(1); list1.add(2); list1.add(3); //3.1 通过已存在的集合创建 ImmutableSet immutableSet = ImmutableSet.copyOf(list1); //3.2 通过初始值,直接创建不可变集合 ImmutableSet immutableSet1 = ImmutableSet.of(1,2,3,4,5,6,7,8,9); //3.3 以builder方式创建 ImmutableSet.builder().add(1).addAll(Sets.newHashSet(1,3)).add(4).build(); //3.4 不支持增删改 //immutableSet.add(8); } public void testCanChange(List<Integer> list1){ list1.remove(0); }
打印日志:
java.lang.UnsupportedOperationException at com.google.common.collect.ImmutableCollection.add(ImmutableCollection.java:244) at com.imooc.zhangxiaoxi.guava.ImmutableTest.testGuavaNoChangeCollect(ImmutableTest.java:77) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:564) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) at org.junit.runners.ParentRunner.run(ParentRunner.java:363) at org.junit.runner.JUnitCore.run(JUnitCore.java:137) at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68) at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47) at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242) at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70) Process finished with exit code -1
guava 依赖
<dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId> <version>28.0-jre</version> </dependency>