Предикат в Java

Я прохожу код, который использует Predicate в Java. Я никогда не использовал Predicate. Может кто-то вести меня к любому учебному или концептуальному объяснению Predicate и его реализация в Java?

100
задан Xaerxess 29 July 2016 в 10:13
поделиться

4 ответа

Я предполагаю, что вы говорите о com.google.common.base.Predicate от Guava.

Из API:

Определяет значение истина или ложь для заданного входа. Например, RegexPredicate может реализовывать Predicate и возвращать истину для любой строки, которая соответствует заданному регулярному выражению.

По сути, это абстракция ООП для логического теста.

Например, у вас может быть вспомогательный метод вроде этого:

static boolean isEven(int num) {
   return (num % 2) == 0; // simple
}

Теперь, имея List , вы можете обрабатывать только четные числа следующим образом:

    List<Integer> numbers = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
    for (int number : numbers) {
        if (isEven(number)) {
            process(number);
        }
    }

With Предикат , тест if абстрагируется как тип. Это позволяет ему взаимодействовать с остальной частью API, такой как Iterables , у которых есть много служебных методов, принимающих Predicate .

Таким образом, теперь вы можете написать что-то вроде этого:

    Predicate<Integer> isEven = new Predicate<Integer>() {
        @Override public boolean apply(Integer number) {
            return (number % 2) == 0;
        }               
    };
    Iterable<Integer> evenNumbers = Iterables.filter(numbers, isEven);

    for (int number : evenNumbers) {
        process(number);
    }

Обратите внимание, что теперь цикл for-each стал намного проще без теста if .Мы достигли более высокого уровня абстракции, определив Iterable evenNumbers , с помощью filter -ing с использованием Predicate .

Ссылки API

  • Iterables.filter
    • Возвращает элементы, удовлетворяющие предикату.

В функции высшего порядка

Предикат позволяет Iterables.filter служить так называемой функцией высшего порядка. Само по себе это дает много преимуществ. Возьмем приведенный выше пример List numbers . Предположим, мы хотим проверить, все ли числа положительны. Мы можем написать что-то вроде этого:

static boolean isAllPositive(Iterable<Integer> numbers) {
    for (Integer number : numbers) {
        if (number < 0) {
            return false;
        }
    }
    return true;
}

//...
if (isAllPositive(numbers)) {
    System.out.println("Yep!");
}

С Predicate и взаимодействуя с остальными библиотеками, мы можем вместо этого написать следующее:

Predicate<Integer> isPositive = new Predicate<Integer>() {
    @Override public boolean apply(Integer number) {
        return number > 0;
    }       
};

//...
if (Iterables.all(numbers, isPositive)) {
    System.out.println("Yep!");
}

Надеюсь, теперь вы можете видеть значение в более высоких абстракциях для подпрограмм например, «фильтровать все элементы по заданному предикату», «проверять, все ли элементы удовлетворяют заданному предикату» и т. д. делают код лучше.

К сожалению, в Java нет первоклассных методов: вы не можете передавать методы в Iterables.filter и Iterables.all . Конечно, вы можете передавать объектов в Java. Таким образом, определяется тип Predicate , и вместо него вы передаете объектов , реализующих этот интерфейс.

См. Также

203
ответ дан 24 November 2019 в 04:50
поделиться

Добавляем к тому, что сказал Майкл :

Вы можете использовать предикат для фильтрации коллекций в java следующим образом:

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

одним из возможных предикатов может быть :

final Predicate<DisplayFieldDto> filterCriteria = 
                    new Predicate<DisplayFieldDto>() {
   public boolean apply(final DisplayFieldDto displayFieldDto) {
    return displayFieldDto.isDisplay();
   }
  };

Использование:

 final List<DisplayFieldDto> filteredList=
 (List<DisplayFieldDto>)filter(displayFieldsList, filterCriteria);
0
ответ дан 24 November 2019 в 04:50
поделиться

Предикат - это функция, которая возвращает значение истина / ложь (т. Е. Логическое), в отличие от предложения, которое является значением истина / ложь (т. Е. Логическим). В Java не может быть автономных функций, поэтому один создает предикат, создавая интерфейс для объекта, который представляет предикат, а затем предоставляет класс, реализующий этот интерфейс. Примером интерфейса для предиката может быть:

public interface Predicate<ARGTYPE>
{
    public boolean evaluate(ARGTYPE arg);
}

И тогда у вас может быть такая реализация, как:

public class Tautology<E> implements Predicate<E>
{
     public boolean evaluate(E arg){
         return true;
     }
}

Чтобы получить лучшее концептуальное понимание, вы можете прочитать о логике первого порядка.

Редактировать
Существует стандартный интерфейс Predicate ( java.util.function.Predicate ), определенный в Java API начиная с Java 8. До Java 8 вы может оказаться удобным повторно использовать интерфейс com.google.common.base.Predicate из Guava .

Также обратите внимание, что начиная с Java 8 намного проще писать предикаты с использованием лямбда-выражений. Например, в Java 8 и выше можно передать p -> true функции вместо определения именованного подкласса Tautology, как указано выше.

14
ответ дан 24 November 2019 в 04:50
поделиться

Вы можете посмотреть java doc примеры или пример использования предиката здесь

В основном он используется для фильтрации строк в наборе результатов на основе каких-либо определенных критериев, которые у вас могут быть, и возвращает true для тех строк, которые соответствуют вашим критериям:

 // the age column to be between 7 and 10
    AgeFilter filter = new AgeFilter(7, 10, 3);

    // set the filter.
    resultset.beforeFirst();
    resultset.setFilter(filter);
0
ответ дан 24 November 2019 в 04:50
поделиться
Другие вопросы по тегам:

Похожие вопросы: