我想要一个过滤器 java.util.Collection 基于谓词。

有帮助吗?

解决方案

Java 8( 2014 )使用Java解决了这个问题一行代码中的流和lambdas:

List<Person> beerDrinkers = persons.stream()
    .filter(p -> p.getAge() > 16).collect(Collectors.toList());

这是教程

使用 Collection#removeIf 来修改集合。 (注意:在这种情况下,谓词将删除满足谓词的对象):

persons.removeIf(p -> p.getAge() <= 16);

lambdaj 允许过滤集合而无需编写循环或内部类:

List<Person> beerDrinkers = select(persons, having(on(Person.class).getAge(),
    greaterThan(16)));

你能想象一些更具可读性的东西吗?

免责声明:我是lambdaj的贡献者

其他提示

假设您使用的是 Java 1.5 ,那你不能添加 Google Collections ,我会做一些非常类似于谷歌人的做法。这是Jon的评论略有不同。

首先将此界面添加到您的代码库中。

public interface IPredicate<T> { boolean apply(T type); }

当某个谓词对某种类型为真时,它的实现者可以回答。例如。如果T UserAuthorizedUserPredicate<User>实现IPredicate<T>,那么AuthorizedUserPredicate#apply将返回传入的get()是否已获得授权。

然后在某个实用程序类中,你可以说

public static <T> Collection<T> filter(Collection<T> target, IPredicate<T> predicate) {
    Collection<T> result = new ArrayList<T>();
    for (T element: target) {
        if (predicate.apply(element)) {
            result.add(element);
        }
    }
    return result;
}

所以,假设您使用上述内容可能

Predicate<User> isAuthorized = new Predicate<User>() {
    public boolean apply(User user) {
        // binds a boolean method in User to a reference
        return user.isAuthorized();
    }
};
// allUsers is a Collection<User>
Collection<User> authorizedUsers = filter(allUsers, isAuthorized);

如果关注线性检查的性能,那么我可能希望拥有一个具有目标集合的域对象。具有目标集合的域对象将具有用于初始化,添加和设置目标集合的方法的过滤逻辑。

更新:

在实用程序类中(比如说Predicate),我在谓词没有返回预期值时添加了一个带有默认值选项的select方法,还添加了一个在新IPredicate中使用的params的静态属性

public class Predicate {
    public static Object predicateParams;

    public static <T> Collection<T> filter(Collection<T> target, IPredicate<T> predicate) {
        Collection<T> result = new ArrayList<T>();
        for (T element : target) {
            if (predicate.apply(element)) {
                result.add(element);
            }
        }
        return result;
    }

    public static <T> T select(Collection<T> target, IPredicate<T> predicate) {
        T result = null;
        for (T element : target) {
            if (!predicate.apply(element))
                continue;
            result = element;
            break;
        }
        return result;
    }

    public static <T> T select(Collection<T> target, IPredicate<T> predicate, T defaultValue) {
        T result = defaultValue;
        for (T element : target) {
            if (!predicate.apply(element))
                continue;
            result = element;
            break;
        }
        return result;
    }
}

以下示例查找集合之间缺少的对象:

List<MyTypeA> missingObjects = (List<MyTypeA>) Predicate.filter(myCollectionOfA,
    new IPredicate<MyTypeA>() {
        public boolean apply(MyTypeA objectOfA) {
            Predicate.predicateParams = objectOfA.getName();
            return Predicate.select(myCollectionB, new IPredicate<MyTypeB>() {
                public boolean apply(MyTypeB objectOfB) {
                    return objectOfB.getName().equals(Predicate.predicateParams.toString());
                }
            }) == null;
        }
    });

以下示例在集合中查找实例,并在找不到实例时将集合的第一个元素作为默认值返回:

MyType myObject = Predicate.select(collectionOfMyType, new IPredicate<MyType>() {
public boolean apply(MyType objectOfMyType) {
    return objectOfMyType.isDefault();
}}, collectionOfMyType.get(0));

更新(在Java 8发布之后):

自从我(艾伦)第一次发布这个答案以来已经好几年了,我仍然无法相信我正在为这个答案收集SO点数。无论如何,现在Java 8已经引入了该语言的闭包,我的答案现在会有很大的不同,而且更简单。使用Java 8,不需要一个独特的静态实用程序类。所以如果你想找到与你的谓词匹配的第一个元素。

final UserService userService = ... // perhaps injected IoC
final Optional<UserModel> userOption = userCollection.stream().filter(u -> {
    boolean isAuthorized = userService.isAuthorized(u);
    return isAuthorized;
}).findFirst();

用于选项的JDK 8 API能够isPresent()orElse(defaultUser)orElseGet(userSupplier)orElseThrow(exceptionSupplier)map,以及其他“monadic”函数,例如flatMapfilterCollectors

如果您只想收集与谓词匹配的所有用户,请使用<=>终止所需集合中的流。

final UserService userService = ... // perhaps injected IoC
final List<UserModel> userOption = userCollection.stream().filter(u -> {
    boolean isAuthorized = userService.isAuthorized(u);
    return isAuthorized;
}).collect(Collectors.toList());

有关如何使用的更多示例,请参见此处 Java 8流工作。

<!>

QUOT;最佳QUOT <!>;方式太宽泛了。它是<!>“最短的<!>”;? <!> QUOT; <!>最快QUOT ;? <!> QUOT; <!>可读QUOT ;? 过滤到另一个集合?

最简单(但不是最可读)的方法是迭代它并使用Iterator.remove()方法:

Iterator<Foo> it = col.iterator();
while( it.hasNext() ) {
  Foo foo = it.next();
  if( !condition(foo) ) it.remove();
}

现在,为了使其更具可读性,您可以将其包装到实用程序方法中。然后发明一个IPredicate接口,创建该接口的匿名实现,并执行以下操作:

CollectionUtils.filterInPlace(col,
  new IPredicate<Foo>(){
    public boolean keepIt(Foo foo) {
      return foo.isBar();
    }
  });

其中filterInPlace()迭代集合并调用Predicate.keepIt()以了解要保存在集合中的实例。

我真的没有理由为这项任务引入第三方库。

考虑 Google Collections 以获取支持通用的更新收藏夹框架。

更新:现在不推荐使用Google集合库。您应该使用最新版本的 Guava 。它仍然具有集合框架的所有相同扩展,包括基于谓词进行过滤的机制。

等待Java 8:

List<Person> olderThan30 = 
  //Create a Stream from the personList
  personList.stream().
  //filter the element to select only those with age >= 30
  filter(p -> p.age >= 30).
  //put those filtered elements into a new List.
  collect(Collectors.toList());

自Java 8早期发布以来,您可以尝试类似:

Collection<T> collection = ...;
Stream<T> stream = collection.stream().filter(...);

例如,如果您有一个整数列表,并且想要过滤<!> gt;的数字; 10然后将这些数字打印到控制台,您可以执行以下操作:

List<Integer> numbers = Arrays.asList(12, 74, 5, 8, 16);
numbers.stream().filter(n -> n > 10).forEach(System.out::println);

我会在铃声中抛出 RxJava ,这也可以在 Android 。 RxJava可能并不总是最佳选择,但如果您希望在集合中添加更多转换或在过滤时处理错误,它将为您提供更大的灵活性。

Observable.from(Arrays.asList(1, 2, 3, 4, 5))
    .filter(new Func1<Integer, Boolean>() {
        public Boolean call(Integer i) {
            return i % 2 != 0;
        }
    })
    .subscribe(new Action1<Integer>() {
        public void call(Integer i) {
            System.out.println(i);
        }
    });

输出:

1
3
5

有关RxJava的filter的更多详细信息,请访问此处

设置:

public interface Predicate<T> {
  public boolean filter(T t);
}

void filterCollection(Collection<T> col, Predicate<T> predicate) {
  for (Iterator i = col.iterator(); i.hasNext();) {
    T obj = i.next();
    if (predicate.filter(obj)) {
      i.remove();
    }
  }
}

用法:

List<MyObject> myList = ...;
filterCollection(myList, new Predicate<MyObject>() {
  public boolean filter(MyObject obj) {
    return obj.shouldFilter();
  }
});

如何简单朴素的Java

 List<Customer> list ...;
 List<Customer> newList = new ArrayList<>();
 for (Customer c : list){
    if (c.getName().equals("dd")) newList.add(c);
 }

简单,易读且易于使用(适用于Android!) 但是如果你使用的是Java 8,你可以用一个很好的方式来实现它:

List<Customer> newList = list.stream().filter(c -> c.getName().equals("dd")).collect(toList());

请注意,toList()是静态导入的

您确定要过滤Collection本身而不是迭代器吗?

参见 org.apache.commons.collections.iterators.FilterIterator

或使用apache commons的第4版 org.apache.commons.collections4.iterators.FilterIterator

让<!>#8217;看看如何过滤内置的JDK列表和 Eclipse的/collections/api/list/MutableList.html“rel =”nofollow noreferrer“> MutableList 收藏(以前 GS收藏)。

List<Integer> jdkList = Arrays.asList(1, 2, 3, 4, 5);
MutableList<Integer> ecList = Lists.mutable.with(1, 2, 3, 4, 5);

如果您想过滤少于3的数字,您会看到以下输出。

List<Integer> selected = Lists.mutable.with(1, 2);
List<Integer> rejected = Lists.mutable.with(3, 4, 5);

这里<!>#8217;如何使用匿名内部类作为Predicate进行过滤。

Predicate<Integer> lessThan3 = new Predicate<Integer>()
{
    public boolean accept(Integer each)
    {
        return each < 3;
    }
};

Assert.assertEquals(selected, Iterate.select(jdkList, lessThan3));

Assert.assertEquals(selected, ecList.select(lessThan3));

以下是使用 Eclipse Collections MutableLists的一些替代方法“https://www.eclipse.org/collections/javadoc/7.0.0/org/eclipse/collections/impl/block/factory/Predicates.html"rel =”nofollow noreferrer“>谓词工厂。

Assert.assertEquals(selected, Iterate.select(jdkList, Predicates.lessThan(3)));

Assert.assertEquals(selected, ecList.select(Predicates.lessThan(3)));

这是一个不为谓词分配对象的版本,使用谓词2 工厂,而不是采用selectWithPredicate2方法。

Assert.assertEquals(
    selected, ecList.selectWith(Predicates2.<Integer>lessThan(), 3));

有时您想要在负面条件下进行过滤。 Eclipse Collections中有一个名为reject的特殊方法。

Assert.assertEquals(rejected, Iterate.reject(jdkList, lessThan3));

Assert.assertEquals(rejected, ecList.reject(lessThan3));

这里<!>#8217;如何使用Java 8 lambda作为partition进行过滤。

Assert.assertEquals(selected, Iterate.select(jdkList, each -> each < 3));
Assert.assertEquals(rejected, Iterate.reject(jdkList, each -> each < 3));

Assert.assertEquals(selected, gscList.select(each -> each < 3));
Assert.assertEquals(rejected, gscList.reject(each -> each < 3));

方法<=>将返回两个集合,其中包含由<=>选择和拒绝的元素。

PartitionIterable<Integer> jdkPartitioned = Iterate.partition(jdkList, lessThan3);
Assert.assertEquals(selected, jdkPartitioned.getSelected());
Assert.assertEquals(rejected, jdkPartitioned.getRejected());

PartitionList<Integer> ecPartitioned = gscList.partition(lessThan3);
Assert.assertEquals(selected, ecPartitioned.getSelected());
Assert.assertEquals(rejected, ecPartitioned.getRejected());

注意:我是Eclipse Collections的提交者。

与ForEach DSL你可以写信

import static ch.akuhn.util.query.Query.select;
import static ch.akuhn.util.query.Query.$result;
import ch.akuhn.util.query.Select;

Collection<String> collection = ...

for (Select<String> each : select(collection)) {
    each.yield = each.value.length() > 3;
}

Collection<String> result = $result();

鉴于收集的[所述,快速、棕色狐狸,跳过的,懒惰的狗]这种结果在[快速的、棕色的跳跃,结束了,懒惰],即所有串超过三个字符。

所有迭代的款式的支持ForEach DSL是

  • AllSatisfy
  • AnySatisfy
  • Collect
  • Counnt
  • CutPieces
  • Detect
  • GroupedBy
  • IndexOf
  • InjectInto
  • Reject
  • Select

欲了解更多详情,请参阅 https://www.iam.unibe.ch/scg/svn_repos/Sources/ForEach

Collections2.filter(Collection,Predicate)方法libraries /“rel =”nofollow noreferrer“> Google的Guava库正是您所需要的。

启用 java 9 Collectors.filtering

public static <T, A, R>
    Collector<T, ?, R> filtering(Predicate<? super T> predicate,
                                 Collector<? super T, A, R> downstream)

因此过滤应该是:

collection.stream().collect(Collectors.filtering(predicate, collector))

示例:

List<Integer> oddNumbers = List.of(1, 19, 15, 10, -10).stream()
            .collect(Collectors.filtering(i -> i % 2 == 1, Collectors.toList()));

这与缺乏真正的闭包相结合,是我对Java最大的抱怨。 老实说,上面提到的大多数方法都很容易阅读,真正有效;然而,在花费时间与.Net,Erlang等之后......在语言层面集成的列表理解使得一切都变得更加清晰。如果没有语言级别的补充,Java就不能像这个领域中的许多其他语言一样干净。

如果性能是一个巨大的问题,谷歌集合是要走的路(或编写自己的简单谓词实用程序)。对于某些人来说,Lambdaj语法更具可读性,但效率并不高。

然后我写了一个图书馆。我会忽略任何有关其效率的问题(是的,它那么糟糕)......是的,我知道它基于清晰的反思,并且我没有实际使用它,但它确实有效:

LinkedList<Person> list = ......
LinkedList<Person> filtered = 
           Query.from(list).where(Condition.ensure("age", Op.GTE, 21));

LinkedList<Person> list = ....
LinkedList<Person> filtered = Query.from(list).where("x => x.age >= 21");

JFilter http://code.google.com/p/jfilter/ 是最适合您的要求。

JFilter是一个简单和高效开放源图书馆来查询的收集Java豆。

关键特征

  • 支持收集(java。工具.收集、java。工具.地图和阵列)的特性。
  • 支持收集的内部收集的任何深度。
  • 支助的内部查询。
  • 支持参数化的询问。
  • 可以过滤1万条记录在几100毫秒。
  • 过滤器(查询)给出简单的关系图来描述软件所需的依赖,它是喜欢Mangodb查询。以下是一些例子。
  • {的"id":{"$le":"10"}
    • 在object id产小于等于10.
  • {的"id":{"$在":["0", "100"]}}
    • 在object id酒店是0或100。
  • {"系列商品":{"lineAmount":"1"}}
    • 在系列商品集合财产的化类型有lineAmount等于1。
  • {"$和":[{的"id":"0"},{"billingAddress":{的"城市":"德"}}]}
    • where id酒店是0和billingAddress.城市的酒店DEL。
  • {"系列商品":{"税":{的"关键":{"代码":"消费税"},"价值":{"$gt":"1.01"}}}}
    • 在系列商品集合财产的参数化的类型,有的税收地图的类型财产的parameteriszed类型的代码等于消费税值大于1.01.
  • {'$或':[{'码':'10'},{'sku':{'$和':[{'价格':{'$在':['20', '40']}}, {'代码':'RedApple'}]}}]}
    • 选择的所有产品在产品代码为10或商品的价格中20、40和商品代码为"RedApple".

我写了扩展的Iterable类,支持应用功能算法而不复制集合内容。

用法:

List<Integer> myList = new ArrayList<Integer>(){ 1, 2, 3, 4, 5 }

Iterable<Integer> filtered = Iterable.wrap(myList).select(new Predicate1<Integer>()
{
    public Boolean call(Integer n) throws FunctionalException
    {
        return n % 2 == 0;
    }
})

for( int n : filtered )
{
    System.out.println(n);
}

上面的代码实际上会执行

for( int n : myList )
{
    if( n % 2 == 0 ) 
    {
        System.out.println(n);
    }
}

使用集合查询引擎(CQEngine)。这是迄今为止最快的方法。

另请参阅:如何在Java中查询对象集合(Criteria / SQL-like)?

这里有一些非常好的答案。我,我想让事情尽可能简单易读:

public abstract class AbstractFilter<T> {

    /**
     * Method that returns whether an item is to be included or not.
     * @param item an item from the given collection.
     * @return true if this item is to be included in the collection, false in case it has to be removed.
     */
    protected abstract boolean excludeItem(T item);

    public void filter(Collection<T> collection) {
        if (CollectionUtils.isNotEmpty(collection)) {
            Iterator<T> iterator = collection.iterator();
            while (iterator.hasNext()) {
                if (excludeItem(iterator.next())) {
                    iterator.remove();
                }
            }
        }
    }
}

简单的Java8之前的解决方案:

ArrayList<Item> filtered = new ArrayList<Item>(); 
for (Item item : items) if (condition(item)) filtered.add(item);

不幸的是,这个解决方案不是完全通用的,输出列表而不是给定集合的类型。此外,引入库或编写包含此代码的函数对我来说似乎有点过头了,除非条件很复杂,但是你可以为条件编写一个函数。

https://code.google.com/p/joquery/

支持不同的可能性,

鉴于收藏,

Collection<Dto> testList = new ArrayList<>();

类型,

class Dto
{
    private int id;
    private String text;

    public int getId()
    {
        return id;
    }

    public int getText()
    {
        return text;
    }
}

过滤

Java 7

Filter<Dto> query = CQ.<Dto>filter(testList)
    .where()
    .property("id").eq().value(1);
Collection<Dto> filtered = query.list();

Java 8

Filter<Dto> query = CQ.<Dto>filter(testList)
    .where()
    .property(Dto::getId)
    .eq().value(1);
Collection<Dto> filtered = query.list();

此外,

Filter<Dto> query = CQ.<Dto>filter()
        .from(testList)
        .where()
        .property(Dto::getId).between().value(1).value(2)
        .and()
        .property(Dto::grtText).in().value(new string[]{"a","b"});

排序(也适用于Java 7)

Filter<Dto> query = CQ.<Dto>filter(testList)
        .orderBy()
        .property(Dto::getId)
        .property(Dto::getName)
    Collection<Dto> sorted = query.list();

分组(也适用于Java 7)

GroupQuery<Integer,Dto> query = CQ.<Dto,Dto>query(testList)
        .group()
        .groupBy(Dto::getId)
    Collection<Grouping<Integer,Dto>> grouped = query.list();

加入(也适用于Java 7)

鉴于,

class LeftDto
{
    private int id;
    private String text;

    public int getId()
    {
        return id;
    }

    public int getText()
    {
        return text;
    }
}

class RightDto
{
    private int id;
    private int leftId;
    private String text;

    public int getId()
    {
        return id;
    }

    public int getLeftId()
        {
            return leftId;
        }

    public int getText()
    {
        return text;
    }
}

class JoinedDto
{
    private int leftId;
    private int rightId;
    private String text;

    public JoinedDto(int leftId,int rightId,String text)
    {
        this.leftId = leftId;
        this.rightId = rightId;
        this.text = text;
    }

    public int getLeftId()
    {
        return leftId;
    }

    public int getRightId()
        {
            return rightId;
        }

    public int getText()
    {
        return text;
    }
}

Collection<LeftDto> leftList = new ArrayList<>();

Collection<RightDto> rightList = new ArrayList<>();

可以加入像,

Collection<JoinedDto> results = CQ.<LeftDto, LeftDto>query().from(leftList)
                .<RightDto, JoinedDto>innerJoin(CQ.<RightDto, RightDto>query().from(rightList))
                .on(LeftFyo::getId, RightDto::getLeftId)
                .transformDirect(selection ->  new JoinedDto(selection.getLeft().getText()
                                                     , selection.getLeft().getId()
                                                     , selection.getRight().getId())
                                 )
                .list();

<强>表达式

Filter<Dto> query = CQ.<Dto>filter()
    .from(testList)
    .where()
    .exec(s -> s.getId() + 1).eq().value(2);

我的回答建立在Kevin Wong的基础之上,这里是使用 spring CollectionUtils和Java 8 lambda 表达式的单行。

CollectionUtils.filter(list, p -> ((Person) p).getAge() > 16);

这与我见过的任何替代方案一样简洁易读(不使用基于方面的库)

Spring CollectionUtils 可从Spring版本4.0.2.RELEASE获得,并记住您需要JDK 1.8和语言级别8 +。

使用java 8,特别是lambda expression,您可以像下面的示例一样:

myProducts.stream().filter(prod -> prod.price>10).collect(Collectors.toList())

对于每个product内部myProducts集合,如果prod.price>10,则将此产品添加到新的过滤列表中。

我需要根据列表中已存在的值过滤列表。例如,删除小于当前值的所有值。 {2 5 3 4 7 5} - <!> gt; {2 5 7}。或者例如删除所有重复项{3 5 4 2 3 5 6} - <!> gt; {3 5 4 2 6}。

public class Filter {
    public static <T> void List(List<T> list, Chooser<T> chooser) {
        List<Integer> toBeRemoved = new ArrayList<>();
        leftloop:
        for (int right = 1; right < list.size(); ++right) {
            for (int left = 0; left < right; ++left) {
                if (toBeRemoved.contains(left)) {
                    continue;
                }
                Keep keep = chooser.choose(list.get(left), list.get(right));
                switch (keep) {
                    case LEFT:
                        toBeRemoved.add(right);
                        continue leftloop;
                    case RIGHT:
                        toBeRemoved.add(left);
                        break;
                    case NONE:
                        toBeRemoved.add(left);
                        toBeRemoved.add(right);
                        continue leftloop;
                }
            }
        }

        Collections.sort(toBeRemoved, new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });

        for (int i : toBeRemoved) {
            if (i >= 0 && i < list.size()) {
                list.remove(i);
            }
        }
    }

    public static <T> void List(List<T> list, Keeper<T> keeper) {
        Iterator<T> iterator = list.iterator();
        while (iterator.hasNext()) {
            if (!keeper.keep(iterator.next())) {
                iterator.remove();
            }
        }
    }

    public interface Keeper<E> {
        boolean keep(E obj);
    }

    public interface Chooser<E> {
        Keep choose(E left, E right);
    }

    public enum Keep {
        LEFT, RIGHT, BOTH, NONE;
    }
}

这将像这样使用。

List<String> names = new ArrayList<>();
names.add("Anders");
names.add("Stefan");
names.add("Anders");
Filter.List(names, new Filter.Chooser<String>() {
    @Override
    public Filter.Keep choose(String left, String right) {
        return left.equals(right) ? Filter.Keep.LEFT : Filter.Keep.BOTH;
    }
});

使用番石榴:

Collection<Integer> collection = Lists.newArrayList(1, 2, 3, 4, 5);

Iterators.removeIf(collection.iterator(), new Predicate<Integer>() {
    @Override
    public boolean apply(Integer i) {
        return i % 2 == 0;
    }
});

System.out.println(collection); // Prints 1, 3, 5

在Java 8中,您可以直接使用此过滤器方法然后执行此操作。

 List<String> lines = Arrays.asList("java", "pramod", "example");

 List<String> result = lines.stream()              
         .filter(line -> !"pramod".equals(line))     
         .collect(Collectors.toList());              

 result.forEach(System.out::println); 
许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top