자바의 function interface
자바로 코딩을 하다보면 자바 라이브러리 함수들에서 아래와 같이 매개변수로 함수를 넣으라는 함수를 마주하게 된다.
// stream map
<R> Stream<R> map(Function<? super T, ? extends R> mapper);
보면 Predicate, Function, Supplier, Consumer, Operator이 있는 것 같은데 이들은 각각 무엇을 의미하는걸까?
(참고로 이들은 java.util.function에 존재한다.)
Predicate<T>
기본 형태는 아래와 같으며 boolean을 반환하는 함수를 의미한다. Predicate<T>를 사용하는 예로 스트림의 filter()를 들 수 있다.
@FunctionalInterface
public interface Predicate<T> {
/**
* Evaluates this predicate on the given argument.
*
* @param t the input argument
* @return {@code true} if the input argument matches the predicate,
* otherwise {@code false}
*/
boolean test(T t);
...
}
아래와 같이 사용되며 Predicate는 하나의 매개변수를 받아서 해당 매개변수를 통해 true/false 값을 도툴한다.
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
list.stream().filter((animal) -> animal.length() > 3);
}
Predicate안에는 and, or 함수가 있어 명시적으로 조건을 and, or로 조합할 수 있다.
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
Predicate<String> predicate1 = (animal) -> animal.length() > 3;
Predicate<String> predicate2 = (animal) -> animal.startsWith("r");
Predicate<String> predicate3 = (animal) -> animal.startsWith("c");
list.stream().filter(predicate1.and(predicate2));
list.stream().filter(predicate1.or(predicate3));
}
Function<T, R>
Function은 매개변수를 하나 받아서 값을 반환하는 함수형 인터페이스이다. 스트임의 map() 함수를 예로 들수 있다.
@FunctionalInterface
public interface Function<T, R> {
/**
* Applies this function to the given argument.
*
* @param t the function argument
* @return the function result
*/
R apply(T t);
...
}
Function은 기본 타입 Double, Int, Long을 타입으로 사용하는 별도 Function들을 제공한다. 여기서 제공하지 않는 타입은 Function<T, R>을 이용해서 정의하면 된다.
예를 들어 스트림 map()을 아래와 같이 사용하던 것을 풀어보면
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
System.out.println(list.stream().map(animal -> animal + "!").collect(Collectors.toList()));
// [cat!, rabbit!, ferret!, dob!]
}
아래와 같이 그 안에서 Function<String, String>을 매개변수로 받아서 사용했다는 것을 알 수 있다.
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
Function<String, String> function = animal -> animal + "!";
System.out.println(list.stream().map(function).collect(Collectors.toList()));
}
Supplier<T>
Supplier<T>는 매개변수가 없이 값을 반환하는 함수형 인터페이스이다.
@FunctionalInterface
public interface Supplier<T> {
/**
* Gets a result.
*
* @return a result
*/
T get();
}
간단히 표현해보면 아래와 같다.
Supplier<String> supplier = () -> { return "Monday!!!!"; };
어떤 값을 반환하느냐에 따라를 여러 가지 Supplier가 존재하지만 이것들은 정확히는 Supplier<T>의 하위클래스들이 아닌 별도의 클래스이다.
BooleanSupplier, DoubleSupplier, IntSupplier, LongSupplier, BooleanSupplier가 존재한다.
그 밖의 타입들은 Supplier<T>를 통해서 표현이 가능하다.
Supplier<T>와 XXXSupplier의 안을 보면 매개변수가 없이 값을 반환하는 함수 하나만 존재한다.
Spplier<T>를 사용하는 방법은 아래와 같다. 단순히 특정 연산을 통해 값을 반환하는 함수를 변수처럼 사용한다고 보면된다. 함수의 반환값 자체를 얻는 것은 Supplier<T>에 해당 T타입으로 반환하는 값으로 반환한도록 정의를 한 후 get()을 사용하면 된다. 물론 T로 정의해놓은 타입이 아닌 다른 타입을 반환하게 하면 Supplier<T>에서 값을 반환하는 부분에서 에러가 발생한다. 컴파일 에러가 발생한다. XXXSupplier는 아예 함수 이름 밑 getXXX()함수로 타입을 정의해놓고 있어서 명확한 기본형을 반환할 때 확실히 할 수 있다.
@Test
public void spplierTest() {
Supplier<String> supplier = () -> { return "Monday!!!!"; };
System.out.println(supplier.get());
}
Consumer<T>
consumer<T>는 매개변수 하나를 받고 아무 값도 반환하지 않는 함수형 인터페이스이다.
@FunctionalInterface
public interface Consumer<T> {
/**
* Performs this operation on the given argument.
*
* @param t the input argument
*/
void accept(T t);
...
}
스트림이나 컬렉션의 forEach에서 Consumer<T>를 받아서 사용한다.
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
list.forEach(animal -> System.out.println(animal) );
}
위 코드를 Consumter<T>로 풀어쓰면 아래와 같다.
@Test
public void spplierTest() {
List<String> list = Arrays.asList("cat", "rabbit", "ferret", "dob");
Consumer<String> consumer = animal -> System.out.println(animal);
list.forEach(consumer);
}
XXXOperator
Operator<T>는 제공하지 않고 XXXOperator들을 제공한다. XXX는 타입이름을 의미하고 Operator는 해당 타입의 값 두 개를 매개변수로 받아서 로직을 수행항 결과 값을 같은 타입으로 반환한다.
@FunctionalInterface
public interface DoubleBinaryOperator {
/**
* Applies this operator to the given operands.
*
* @param left the first operand
* @param right the second operand
* @return the operator result
*/
double applyAsDouble(double left, double right);
}
XXXUnaryOperator도 있는데 이 함수형 인터페이스들은 하나의 해당 타입의 값 한 개를 매개변수로 받아서 로직을 수행한 결과 값을 같은 타입으로 반환한다.
@FunctionalInterface
public interface DoubleUnaryOperator {
/**
* Applies this operator to the given operand.
*
* @param operand the operand
* @return the operator result
*/
double applyAsDouble(double operand);
...
}
function interface는 언제 사용하는가?
우선 위의 코드들을 보면 매개변수로 함수를 넘겨야할 때 함수형 인터페이스들을 사용했다. 기본적으로 함수형 인터페이스를 넘기는 것은 어떤 기능을 할지 모르는 함수가 넘어올 수 있어 사용을 지향하는 것이 좋지만 자바 라이브러리 같은 경우는 함수형 인터페이스를 넘겨 받아서 실행을 시키도록 되어 있다. 이럴 경우는 함수형 인터페이스를 사용해야한다.
그렇다면 그런 경우가 아니라면 함수형 인터페이스를 사용할 필요가 있을까라고 묻는다면 코드의 간결성을 이야기할 수 있다. 함수형 인터페이스를 매개변수로 필요로 하는 함수 말고도 특히 Supplier<T>의 경우는 실행 시점에 따라 값이 다르게 반환되는 경우 유용하게 사용될 수 있다. 그렇다면 이를 함수로 만들어도 되지 않을까? 싶기도 한데 복잡하고 긴 로직이 아닐 경우 함수로 만드는 대신 Supplier<T>로 만드는게 코드를 좀 더 간경하게 만들 수 있다.
@Test
public void spplierTest() {
Supplier<Long> supplier = () -> { return System.currentTimeMillis() % 2; };
System.out.println(supplier.get()); // 1
System.out.println(supplier.get()); // 0
}
✋ Predicate
https://docs.oracle.com/javase/8/docs/api/java/util/function/Predicate.html
✋ Function
https://docs.oracle.com/javase/8/docs/api/java/util/function/Function.html
✋ Supplier
https://docs.oracle.com/javase/8/docs/api/java/util/function/Supplier.html
✋ Consumer
https://docs.oracle.com/javase/8/docs/api/java/util/function/Consumer.html
✋ BinaryOperator<T>
https://docs.oracle.com/javase/8/docs/api/java/util/function/BinaryOperator.html
✋ UnaryOperator<T>
https://docs.oracle.com/javase/8/docs/api/java/util/function/UnaryOperator.html