sci检索没有馆藏号
您是否曾经想替换过HashSet
或HashMap
使用的equals
和hashCode
方法? 或者有一个List
的一些元素类型伪装成的List
相关类型的?
转换集合使这成为可能,并且本文将展示如何实现。
总览
转换集合是LibFX 0.3.0的一项功能,该功能将在每天的今天发布。 这篇文章将介绍总体思路,涵盖技术细节并完成一些可能会派上用场的用例。
正在进行的示例是LibFX中包含的功能演示的稍微改编的变体。 请记住,这只是演示该概念的一个示例。
转变馆藏
转换集合是另一个集合的视图(例如,列表到列表,地图到地图等),它似乎包含不同类型的元素(例如,整数而不是字符串)。
通过应用转换从内部元素创建视图元素。 这是按需发生的,因此转换集合本身是无状态的。 作为一个适当的视图,内部集合以及转换视图的所有更改都将反映在另一个视图中(例如Map及其entrySet )。
命名法
转换集合也可以视为装饰器。 我将装饰后的集合称为内部集合,并将其泛型称为内部类型。 转换集合及其通用类型分别称为外部集合和外部类型。
例
让我们来看一个例子。 假设我们有一组字符串,但是我们知道这些字符串只包含自然数。 我们可以使用一个转换集来获取一个看起来像是整数集的视图。
(类似// "[0, 1] ~ [0, 1]"
的System.out.println(innerSet + " ~ " + transformingSet);
是System.out.println(innerSet + " ~ " + transformingSet);
的控制台输出。)
Set<String> innerSet = new HashSet<>();
Set<Integer> transformingSet = new TransformingSet<>(innerSet,/* skipping some details */);
// both sets are initially empty: "[] ~ []"// now let's add some elements to the inner set
innerSet.add("0");
innerSet.add("1");
innerSet.add("2");
// these elements can be found in the view: "[0, 1, 2] ~ [0, 1, 2]"// modifying the view reflects on the inner set
transformingSet.remove(1);
// again, the mutation is visible in both sets: "[0, 2] ~ [0, 2]"
看看转换有多愉快?
细节
像往常一样,魔鬼在细节中,所以让我们讨论这个抽象的重要部分。
转寄
转换集合是另一个集合的视图。 这意味着它们本身不保存任何元素,而是将所有调用转发给内部/装饰的集合。
他们通过将调用参数从外部类型转换为内部类型并使用这些参数调用内部集合来实现此目的。 然后,将返回值从内部类型转换为外部类型。 对于以集合为参数的调用,这变得有些复杂,但是方法基本上是相同的。
所有转换集合的实现方式都是将方法的每个调用转发到内部集合上的相同方法 (包括default方法 )。 这意味着内部集合对线程安全性,原子性等的任何保证也将由转换集合维护。
转型
转换是通过在构造过程中指定的一对函数来计算的。 一个用于将外部元素转换为内部元素,另一个用于另一个方向。 (对于映射,存在两对这样的对:一对用于键,一对用于值。)
转换函数关于equals
必须彼此相反,即, outer.equals(toOuter(toInner(outer))
和inner.equals(toInner(toOuter(inner))
对于所有外部元素和内部元素必须为true。事实并非如此,这些集合的行为可能无法预测。
对于身份而言,情况并非如此,即, outer == toOuter(toInner(outer))
可能为false。 详细信息取决于所应用的转换,并且通常未指定-它可能永远不会,有时或永远是正确的。
例
让我们看看转换函数如何查找我们的字符串和整数集:
private Integer stringToInteger(String string) {return Integer.parseInt(string);
}private String integerToString(Integer integer) {return integer.toString();
}
这就是我们使用它们创建转换集的方式:
Set<Integer> transformingSet = new TransformingSet<>(innerSet,this::stringToInteger, this::integerToString,/* still skipping some details */);
直截了当吧?
是的,但是即使这个简单的示例也包含陷阱。 注意前导零的字符串如何映射到相同的整数。 这可以用于创建不良行为:
innerSet.add("010");
innerSet.add("10");
// now the transforming sets contains the same entry twice:
// "[010, 10] ~ [10, 10]"// sizes of different sets:
System.out.println(innerSet.size()); // "2"
System.out.println(transformingSet.size()); // "2"
System.out.println(new HashSet<>(transformingSet).size()); // "1" !// removing is also problematic
transformingSet.remove(10) // the call returns true
// one of the elements could be removed: "[010] ~ [10]"
transformingSet.remove(10) // the call returns false
// indeed, nothing changed: "[010] ~ [10]"// now things are crazy - this returns false:
transformingSet.contains(transformingSet.iterator().next())
// the transforming set does not contain its own elements ~> WAT?
因此,在使用转换集合时,仔细考虑转换非常重要。 它们必须彼此相反!
但这仅限于实际发生的内部和外部元素就足够了。 在示例中,问题仅在引入带有前导零的字符串时开始。 如果某些业务规则禁止了这些规则,并且这些规则已经正确执行,那么一切都会好起来的。
类型安全
以通常的静态,编译时方式对转换集合进行的所有操作都是类型安全的。 但是,由于collection接口中的许多方法都允许对象(例如Collection.contains(Object)
)或未知通用类型的集合(例如Collection.addAll(Collection<?>)
)作为参数,因此这并不涵盖所有可能发生在以下情况的情况运行。
请注意,这些调用的参数必须从外部类型转换为内部类型,才能将调用转发到内部集合。 如果使用非外部类型的实例调用它们,则很可能无法将其传递给转换函数。 在这种情况下,该方法可能会抛出ClassCastException
。 尽管这与方法的合同一致,但可能仍然是意外的。
为了降低这种风险,转换集合的构造函数需要使用内部和外部类型的令牌。 它们用于检查元素是否为必需类型,如果不是,则可以毫无例外地优雅地回答查询。
例
我们终于可以确切地看到如何创建转换集:
Set<Integer> transformingSet = new TransformingSet<>(innerSet,String.class, this::stringToInteger,Integer.class, this::integerToString);
构造函数实际上接受Class<? super I>
Class<? super I>
所以这也将编译:
Set<Integer> transformingSetWithoutTokens = new TransformingSet<>(innerSet,Object.class, this::stringToInteger,Object.class, this::integerToString);
但是由于一切都是对象,因此对令牌的类型检查变得无用,并且调用转换函数可能会导致异常:
Object o = new Object();
innerSet.contains(o); // false
transformingSet.contains(o); // false
transformingSetWithoutTokens.contains(o); // exception
用例
我要说的是,转换集合是一种非常专业的工具,不太可能经常使用,但在每个分类良好的工具箱中仍然占有一席之地。
重要的是要注意,如果性能至关重要,则可能会出现问题。 每次调用包含或返回元素的转换集合,都会导致至少创建一个(通常是多个)对象。 这些对垃圾收集器施加了压力,并导致通往有效负载的方式的间接级别更高。 (与以往一样,在讨论性能时:首先介绍个人资料!)
那么转换集合的用例是什么? 上面我们已经看到了如何将集合的元素类型更改为另一种。 虽然这代表了总体思路,但我认为这不是一个非常普遍的用例(尽管在某些边缘情况下是有效的方法)。
在这里,我将展示两个更狭窄的解决方案,您可能希望在某些时候使用它们。 但是我也希望这能使您了解如何使用转换集合来解决棘手的情况。 也许您的问题的解决方案在于巧妙地应用此概念。
用Equals和HashCode代替
我一直很喜欢.NET的哈希图(他们称其为字典)如何具有将EqualityComparer作为参数的构造函数 。 通常将在键上调用的所有equals
和hashCode
调用均委派给该实例。 因此有可能即时替换有问题的实现。
当您处理无法完全控制的有问题的旧版代码或库代码时,这可以节省生命。 当需要一些特殊的比较机制时,它也很有用。
通过转换集合,这很容易。 为了使它更加容易,LibFX已经包含一个EqualityTransformingSet
和EqualityTransformingMap
。 它们修饰另一个集合或映射的实现,并且可以在构造过程中提供键/元素的equals
和hashCode
函数。
例
假设您想将字符串用作set元素,但为了进行比较,您仅对它们的长度感兴趣。
Set<String> lengthSet = EqualityTransformingSet.withElementType(String.class).withInnerSet(new HashSet<Object>()).withEquals((a, b) -> a.length != b.length).withHash(String::length).build();lengthSet.add("a");
lengthSet.add("b");
System.out.println(lengthSet); // "[a]"
从集合中删除可选性
也许您正在与一个想到在各处使用Optional
的想法的人一起工作,然后疯狂地使用它,现在您有了Set<Optional<String>>
。 如果无法修改代码(或您的同事),则可以使用转换集合来获取一个对您隐藏Optional
的视图。
同样,实现起来很简单,因此LibFX已经以OptionalTransforming[Collection|List|Set]
的形式包含了它。
例
Set<Optional<String>> innerSet = new HashSet<>();
Set<String> transformingSet =new OptionalTransformingSet<String>(innerSet, String.class);innerSet.add(Optional.empty());
innerSet.add(Optional.of("A"));// "[Optional.empty, Optional[A]] ~ [null, A]"
请注意, null
表示空的optional的方式。 这是默认行为,但是您也可以将另一个字符串指定为空可选值的值:
Set<String> transformingSet =new OptionalTransformingSet<String>(innerSet, String.class, "DEFAULT");// ... code as above ...
// "[Optional.empty, Optional[A]] ~ [DEFAULT, A]"
这样可以避免Optional以及null作为元素,但是现在您必须确保永远不会有包含DEFAULT的Optional。 (如果确实如此,则隐式转换不是彼此相反的,这在上面已经看到导致问题了。)
有关此示例的更多详细信息,请查看演示 。
反射
我们已经介绍过,转换集合是另一个集合的视图。 使用类型标记(以最大程度地减少ClassCastExceptions
)和一对转换函数(它们必须彼此相反),每个调用都将转发到装饰的集合。 转换后的集合可以维护修饰后的集合所做的关于线程安全性,原子性的所有保证。
然后,我们看到了转换集合的两个特定用例:替换等于和哈希数据结构使用的哈希码,以及从Collection<Optional<E>>
删除可选性。
谈谈LibFX
就像我说的那样,转换集合是我的开源项目LibFX的一部分。 如果您考虑使用它,我想指出一些事情:
- 这篇文章介绍了这个想法和一些细节,但是并不能代替文档。 查看Wiki,获取最新描述和指向Javadoc的指针。
- 我认真对待测试。 多亏了Guava ,约6.500个单元测试涵盖了转换集合。
- LibFX是根据GPL许可的。 如果那不适合您的许可模式,请随时与我联系。
翻译自: https://www.javacodegeeks.com/2015/05/transforming-collections.html
sci检索没有馆藏号