我想检查一个列表是否包含一个对象,该对象具有具有特定值的字段。现在,我可以使用循环来遍历和检查,但我很好奇是否有更高效的代码。
类似的;
if(list.contains(new Object().setName("John"))){
//Do some stuff
}
我知道上面的代码没有做任何事情,它只是粗略地演示了我想要实现的目标。
另外,澄清一下,我不想使用简单循环的原因是,这段代码将进入一个循环,这个循环在一个循环中,而这个循环又在一个循环中。为了可读性,我不想一直在这些循环中添加循环。所以我想知道是否有什么简单的替代方案。
流
如果你正在使用Java 8,也许你可以尝试这样做:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().filter(o -> o.getName().equals(name)).findFirst().isPresent();
}
或者,你可以尝试这样做:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().map(MyObject::getName).filter(name::equals).findFirst().isPresent();
}
如果List<MyObject>包含名称为name的MyObject,此方法将返回true。如果你想对getName().equals(name)的每个myobject执行一个操作,那么你可以尝试这样做:
public void perform(final List<MyObject> list, final String name){
list.stream().filter(o -> o.getName().equals(name)).forEach(
o -> {
//...
}
);
}
其中o表示MyObject实例。
或者,正如评论所建议的(感谢MK10),你可以使用Stream#anyMatch方法:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().anyMatch(o -> name.equals(o.getName()));
}
流
如果你正在使用Java 8,也许你可以尝试这样做:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().filter(o -> o.getName().equals(name)).findFirst().isPresent();
}
或者,你可以尝试这样做:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().map(MyObject::getName).filter(name::equals).findFirst().isPresent();
}
如果List<MyObject>包含名称为name的MyObject,此方法将返回true。如果你想对getName().equals(name)的每个myobject执行一个操作,那么你可以尝试这样做:
public void perform(final List<MyObject> list, final String name){
list.stream().filter(o -> o.getName().equals(name)).forEach(
o -> {
//...
}
);
}
其中o表示MyObject实例。
或者,正如评论所建议的(感谢MK10),你可以使用Stream#anyMatch方法:
public boolean containsName(final List<MyObject> list, final String name){
return list.stream().anyMatch(o -> name.equals(o.getName()));
}
二分查找
您可以使用集合。binarySearch用于搜索列表中的元素(假设列表是有序的):
Collections.binarySearch(list, new YourObject("a1", "b",
"c"), new Comparator<YourObject>() {
@Override
public int compare(YourObject o1, YourObject o2) {
return o1.getName().compareTo(o2.getName());
}
});
如果该对象不存在于集合中,它将返回一个负数,否则将返回该对象的索引。有了它,您可以使用不同的搜索策略搜索对象。
谓词
如果你不使用Java 8,或者不使用为处理集合提供更多功能的库,你可以实现一些比你的解决方案更可重用的东西。
interface Predicate<T>{
boolean contains(T item);
}
static class CollectionUtil{
public static <T> T find(final Collection<T> collection,final Predicate<T> predicate){
for (T item : collection){
if (predicate.contains(item)){
return item;
}
}
return null;
}
// and many more methods to deal with collection
}
我使用类似的东西,我有谓词接口,我把它的实现传递给我的util类。
用我的方式做这件事有什么好处?有一个方法可以处理任何类型集合中的搜索。如果你想通过不同的字段进行搜索,你不必创建单独的方法。你所需要做的就是提供不同的谓词,一旦它不再有用/就可以销毁
如果你想使用它,你所需要做的就是调用方法并定义你的谓词
CollectionUtil.find(list, new Predicate<MyObject>{
public boolean contains(T item){
return "John".equals(item.getName());
}
});