Ответ 1
Я предполагаю, что вы говорите com.google.common.base.Predicate<T>
от Guava.
Из API:
Определяет значение
true
илиfalse
для заданного ввода. Например,RegexPredicate
может реализоватьPredicate<String>
и вернуть значение true для любой строки, которая соответствует заданному регулярному выражению.
Это, по сути, абстракция ООП для теста boolean
.
Например, у вас может быть вспомогательный метод, подобный этому:
static boolean isEven(int num) {
return (num % 2) == 0; // simple
}
Теперь, учитывая a 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
-ing, используя 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
определен, и вместо этого вы передаете объекты, реализующие этот интерфейс.