it-roy-ru.com

Предикат в Java

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

97
srikanth

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

Из API:

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

По сути, это абстракция OOP для теста 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 определен, и вы передаете объекты , реализующие этот интерфейс.

Смотрите также

201
polygenelubricants

Предикат - это функция, которая возвращает значение истина/ложь (то есть логическое), в отличие от предложения, которое является значением истина/ложь (то есть логическое значение). В 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 ), определенный в API Java с Java 8. До Java 8 может оказаться удобным повторно использовать интерфейс com.google.common.base.Predicate из Guava знак равно.

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

14
Michael Aaron Safyan

Вы можете просмотреть 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);
0
techzen

Добавим к тому, что Micheal сказал:

Вы можете использовать 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
jai