Я прохожу код, который использует Predicate
в Java. Я никогда не использовал Predicate
. Может кто-то вести меня к любому учебному или концептуальному объяснению Predicate
и его реализация в Java?
Я предполагаю, что вы говорите о 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
, с помощью filter
-ing с использованием Predicate
.
Iterables.filter
Предикат
позволяет Iterables.filter
служить так называемой функцией высшего порядка. Само по себе это дает много преимуществ. Возьмем приведенный выше пример List
. Предположим, мы хотим проверить, все ли числа положительны. Мы можем написать что-то вроде этого:
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
, и вместо него вы передаете объектов , реализующих этот интерфейс.
Добавляем к тому, что сказал Майкл :
Вы можете использовать предикат для фильтрации коллекций в 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);
Предикат - это функция, которая возвращает значение истина / ложь (т. Е. Логическое), в отличие от предложения, которое является значением истина / ложь (т. Е. Логическим). В 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, как указано выше.
Вы можете посмотреть 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);