Предикат в Java - PullRequest
       27

Предикат в Java

97 голосов
/ 02 июня 2010

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

Ответы [ 4 ]

201 голосов
/ 02 июня 2010

Полагаю, вы говорите о com.google.common.base.Predicate<T> из Гуавы.

Из API:

* +1007 *

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

Это, по сути, абстракция ООП для теста boolean.

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

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

Теперь, учитывая List<Integer>, вы можете обрабатывать только четные числа, как это:

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

При Predicate тест 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<Integer> evenNumbers, filter, используя Predicate.

API ссылки

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

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

Predicate позволяет Iterables.filter служить в качестве так называемой функции высшего порядка. Само по себе это дает много преимуществ. Возьмите List<Integer> 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 определен, и вы передаете объекты , реализующие этот интерфейс.

Смотри также

14 голосов
/ 02 июня 2010

Предикат - это функция, которая возвращает значение истина / ложь (то есть логическое), в отличие от предложения, которое является значением истина / ложь (то есть логическое значение). В 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 интерфейс от Гуава .

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

0 голосов
/ 02 июня 2010

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

Вы можете использовать Predicate следующим образом при фильтрации коллекций в 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 голосов
/ 02 июня 2010

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

Обычно он используется для фильтрации строк в наборе результатов на основе любых конкретных критериев, которые могут у вас быть, и возвращает значение 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);
...