- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
在 List.of() or Collections.emptyList() 中给出的评论和答案的上下文中和 List.of(...) or Collections.unmodifiableList()我提出了以下两个经验法则(相应地也适用于 Set
和 Map
工厂)。
Collections.emptyList()
为了可读性和当例如初始化惰性字段成员,如:
class Bean {
private List<Bean> beans = Collection.emptyList();
public List<Bean> getBeans() {
if (beans == Collections.EMPTY_LIST) { beans = new ArrayList<>(); }
return beans;
}
}
List.of()
当使用
List
调用可执行文件时,和变体作为快速和较少类型的版本参数。这是我目前的替换模式:
Collections.emptyList() --> List.of()
Collections.singletonList(a) --> List.of(a)
Arrays.asList(a, ..., z) --> List.of(a, ..., z)
Collections.indexOfSubList
的虚构用法中,以下几行
Collections.indexOfSubList(Arrays.asList(1, 2, 3), Collections.emptyList());
Collections.indexOfSubList(Arrays.asList(1, 2, 3), Collections.singletonList(1));
Collections.indexOfSubList(Arrays.asList(1, 2, 3), Arrays.asList(1));
Collections.indexOfSubList(Arrays.asList(1, 2, 3), Arrays.asList(2, 3));
Collections.indexOfSubList(Arrays.asList(1, 2, 3), Arrays.asList(1, 2, 3));
Collections.indexOfSubList(List.of(1, 2, 3), List.of());
Collections.indexOfSubList(List.of(1, 2, 3), List.of(1));
Collections.indexOfSubList(List.of(1, 2, 3), List.of(1));
Collections.indexOfSubList(List.of(1, 2, 3), List.of(2, 3));
Collections.indexOfSubList(List.of(1, 2, 3), List.of(1, 2, 3));
最佳答案
一般来说,新工厂的使用对于新代码是安全的,其中没有依赖于现有集合行为的现有代码。
新的集合工厂不能直接替代使用现有 API 初始化集合的代码有几个原因。显然,不变性是最突出的原因之一;如果以后需要修改集合,显然不能一成不变!但也有其他原因,其中一些非常微妙。
有关使用新 API 替换现有 API 的示例,请参阅 JDK-8134373 .评论主题在这里:Part1 Part2 .
这是问题的概述。
数组包装与复制。 有时你有一个数组,例如varargs 参数,并且您希望将其作为列表处理。有时 Arrays.asList
是这里最合适的东西,因为它只是一个包装器。相比之下,List.of
创建一个副本,这可能会很浪费。另一方面,调用者仍然拥有包装数组的句柄并且可以修改它,这可能是一个问题,因此有时您需要支付复制它的费用,例如,如果您想保留对实例变量中的列表。
散列集合迭代顺序。 新款Set.of
和 Map.of
结构随机化它们的迭代顺序。 HashSet
的迭代顺序和 HashMap
是未定义的,但实际上它是相对稳定的。代码可能会无意中产生对迭代顺序的依赖。切换到新的集合工厂可能会将旧代码暴露给迭代顺序依赖性,从而暴露潜在的错误。
禁止空值。 新集合完全禁止空值,而常见的非并发集合( ArrayList
、 HashMap
)允许它们。
序列化格式。 新集合具有与旧集合不同的序列化格式。如果集合被序列化,或者它存储在其他序列化的类中,则序列化输出将有所不同。这可能是也可能不是问题。但是,如果您希望与其他系统互操作,这可能是一个问题。特别是,如果您将新集合的序列化形式传输到 Java 8 JVM,它将无法反序列化,因为 Java 8 上不存在新类。
严格的 Mutator 方法行为。 新集合是不可变的,所以它们当然会抛出 UnsupportedOperationException
当调用mutator方法时。但是,存在一些边缘情况,其中所有集合的行为不一致。例如,
Collections.singletonList("").addAll(Collections.emptyList())
List.of("").addAll(Collections.emptyList())
Collections.empty*
的集合和
Collections.singleton*
, 只有在实际发生突变时才会抛出 UOE。
Set
和
Map
工厂拒绝重复的元素和键。如果您使用常量列表初始化集合,这通常不是问题。实际上,如果常量列表有重复项,则可能是错误。这可能是一个问题,当允许调用者传入元素的集合或数组(例如,可变参数)时。如果调用者传入重复项,现有 API 将忽略重复项,而新工厂将抛出
IllegalArgumentException
.这是一种可能会影响调用者的行为变化。
关于java - Java 9 集合工厂的使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40888754/
我应该执行以下操作: 可能通过服务/工厂,使用 $q(异步)查询 API 以获取大型名称数据集 有另一个服务(也是异步的),它应该只返回上述工厂的元素,如果它们与某个字符串(搜索字段)匹配。目的是缩小
我有一个通用的基类。我有一个实现基类的具体类。 我将如何创建工厂类/方法来交付不同类型的具体类? 举个例子: public class ReceiverBase where T : IInte
我正在查看以下链接中的 Ninject Factory 扩展: http://www.planetgeek.ch/2011/12/31/ninject-extensions-factory-intro
工厂、提供商和服务这三个术语之间有什么区别? 刚刚了解 NHibernate 及其存储库模式(POCO 类等)。 最佳答案 工厂:通过将一堆位组合在一起或基于某种上下文选择类型来组装类 Provide
使用CGLIB我可以做到 final var enhancer = new Enhancer(); enhancer.setUseCache(false); enhancer.setSuperclas
我试图在 Kotlin 中使用伴随对象工厂方法(相当于 Java 中的静态工厂方法)创建一个嵌套内部类。这是我的代码的简化版本。 class OuterClass { var myData:L
我正在为我的大学做一个项目,但遇到了问题。 基本上,该项目由一个客户端-服务器应用程序组成,我想创建一个用于通信的 Packet 类。数据包由 header 和主体组成。现在问题来了。我可以有一些不同
这个问题在这里已经有了答案: Why doesn't polymorphism work without pointers/references? (6 个答案) What is object sl
我正在制作一个套接字工厂。我希望每个外部应用程序都使用 Socket 类的接口(interface),它是几个类(ServerSocketTCP、ClientSocketTCP、ServerSocke
我是 angularjs 的新手,我正在尝试创建一个小型电影数据库。这是我第一次使用工厂,我想确保这是正确的方法,以及如何在另一个功能中使用这个工厂,如下所示? 我希望这个工厂只运行一次,这样我就可以
这个问题在这里已经有了答案: Java inner class and static nested class (28 个答案) 关闭 5 年前。 public class DataFactory
我看过很多关于 C++ 工厂的帖子,但到目前为止我还没有看到解决我的问题的解决方案。 (虽然我可能遗漏了一些东西。) 示例控制台应用程序: #include #include #include
这是一个简单的 C++ 项目,有 2 种设计模式:单例和工厂,sigleton 也是一个模板化类,一个接口(interface) (IHash) 和一个类 (Hash1)。一个简单的工厂类 (Hash
这个问题类似于Factory and generics ,并且可能有相同的答案,但它是不同的。我有一个通用基类,它将由完全独立的 JAR 中的类进行扩展。所述 JAR 应该能够在不更改任何其他代码的情
问题是我需要为传递的类创建一个新实例 有没有办法重写这个函数,让它可以接受任意数量的参数? function createInstance(ofClass, arg1, arg2, arg3, ...
我想用简单的 C++ 语法创建一个简单的工厂方法: void *createObject(const char *str,...) { if(!strcmp("X",str)) retu
经过大约 10 个月的程序化 PHP 学习后,我现在正尝试着手研究基本的 OOP 原则和设计模式。这是一个爱好,我没有那么多时间去追求它,所以请原谅这个问题的水平很低。 我的网站(目前 100% 程序
我有一个简单的问题。 我如何编写一个工厂来定义使用 make() 或 create() 的关系,具体取决于原始调用 make() 还是 create()? 这是我的用例: 我有一个简单的工厂 /**
我正在尝试在延迟加载模块中提供 APP_BASE_HREF 注入(inject) token ,然而,工厂方法根本没有被调用。 在这里https://github.com/MaurizioCascia
我有以下 ast: import { factory as f } from 'typescript' const typeDeclaration = f.createTypeAliasDeclara
我是一名优秀的程序员,十分优秀!