- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有三个接口(interface):
public interface Combinable<V> {
V add(V other);
}
public interface Sublistable<V> {
boolean hasSublist();
List<V> getSublist();
void setSublist(List<V> sublist);
}
public interface HasUniqueIdentifier {
String getUniqueIdentifier();
}
和实现部分或全部这些接口(interface)的 4 个类:
public class Grandparent implements HasUniqueIdentifier,
Sublistable<Parent>,
Combinable<Grandparent>
{ List<Parent> sublist; ... }
public class Parent implements HasUniqueIdentifier,
Sublistable<Child>,
Combinable<Parent>
{ List<Child> sublist; ... }
public class Child implements HasUniqueIdentifier,
Sublistable<Grandchild>,
Combinable<Child>
{ List<Grandchild> sublist; ... }
public class Grandchild implements HasUniqueIdentifier,
Combinable<Grandchild>
{ }
我想要一个执行以下操作的通用方法:
public <V, T extends HasUniqueIdentifier & Combinable<T> & Sublistable<V>>
List<T> combine(List<T> items) {
Multimap<String, T> similarItemMap = HashMultimap.create();
for (T item: items) {
similarItemMap.put(item.getUniqueIdentifier(), item);
}
List<T> output = new ArrayList<T>();
for (Collection<T> similarCollection : similarItemMap.asMap().values()) {
List<T> similarItems = Lists.newArrayList(similarCollection);
T source = similarItems.get(0);
for (int i = 0; i < similarItems.size(); i++) {
source = source.add(similarItems.get(i));
}
output.add(source);
}
for (T item : output) {
if (item.hasSublist()) {
item.setSublist(combine(item.getSublist));
}
}
return output;
}
意识到这可能会造成无限循环(除非底层类——Grandchild——实现了 Sublistable 并设置了 hasSublist() { return false; }
或其他),以及事实这个方法对泛型有点疯狂,我不得不问:有什么方法可以稍微重写一下,以便我可以用这种方式调用方法:
合并(祖 parent 列表)
或者我应该放弃这是一种方法并尝试以更好的方式重构它吗?
编辑:为了更好地解释我正在尝试做什么,我有一个 A
类型的对象列表。每个对象 a
都有一个类型为 B
的对象列表。每个对象 b
都有一个类型为 C
的对象列表,依此类推,直到最终类型为 T
(对于某些不同级别的 T
) 不再有子列表。
每种类型基本上都需要为“合并”或“组合”方法做三件事:
item.getUniqueIdentifier()
方法将所有“喜欢”的项目收集到一个容器中source.add(other)
方法将所有类似的项目组合成一个项目由于每个项目的行为都非常相似,如果我可以使用单一方法而不是必须有 n
方法,每种类型一个方法,那就太好了。不幸的是,由于没有一种类型可以保证是相同的(除了已经实现了上面的部分或全部给定接口(interface)),因此创建一个通用方法被证明是困难的。有什么方法可以做到这一点我想念的吗?
public <V extends HasUniqueIdentifier & Combinable<V>,
T extends HasUniqueIdentifier & Combinable<T>>
List<T> combine(List<T> items) {
Multimap<String, T> similarItemMap = HashMultimap.create();
for (T item: items) {
similarItemMap.put(item.getUniqueIdentifier(), item);
}
List<T> output = new ArrayList<T>();
for (Collection<T> similarCollection : similarItemMap.asMap().values()) {
List<T> similarItems = Lists.newArrayList(similarCollection);
T source = similarItems.get(0);
for (int i = 0; i < similarItems.size(); i++) {
source = source.add(similarItems.get(i));
}
output.add(source);
}
for (T item : output) {
if (item instanceof Sublistable<?>) {
@SuppressWarnings("unchecked")
Sublistable<V> sublistableItem = ((Sublistable<V>)sublistableItem);
if (sublistableItem.hasSublist()) {
sublistableItem.setSublist(combine(sublistableItem.getSublist));
}
}
}
return output;
}
不幸的是,此方法同时需要 @SupressWarnings
和 instanceof
,我希望尽可能避免。不过,我还没有找到其他任何东西。
最佳答案
鉴于 GrandChild implements Sublistable<Void>
,您的第一个解决方案很好空洞地。树更统一,更容易递归处理。您可能还想合并一个 interface Node<T,V> extends HasUniqueIdentifier, Combinable<T>, Sublistable<V>{}
如果你不想GrandChild implements Sublistable
,您的第二个解决方案也可以。单instanceof
测试标记接口(interface)不是罪过。代码可以重写为
public <T extends HasUniqueIdentifier & Combinable<T>>
List<T> combine(List<T> items)
{
...
List<T> output = new ArrayList<T>();
...
for (T item : output) {
if (item instanceof Sublistable<?>)
combineSublist((Sublistable<?>)item);
return output;
}
private <V> void combineSublist(Sublistable<V> sublistableItem)
{
if (sublistableItem.hasSublist()) {
sublistableItem.setSublist(combine(sublistableItem.getSublist));
}
关于java - 使用 Java 泛型通过对象列表递归对象(每个对象都有另一个对象列表),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15241575/
我是一名优秀的程序员,十分优秀!