Functional Interface
Java8 부터 도입된 함수형 인터페이스와 자바에서 기본 제공하는 함수를 알아보겠습니다.
1. @FunctionalInterface
@FunctionalInterface // 함수형 인터페이스 정의 시 견고한 관리 가능
public interface RunSomething {
void doIt();
static void printName() {
System.out.println("Devlee");
}
default void printAge() {
System.out.println("3");
}
}
함수형 인터페이스는 하나의 추상 메서드(SAM : Single Abstract Method)를 가지고 있는 인터페이스를 말합니다. 추가로 Java8 부터 인터페이스는 static
메서드와 default
메서드를 포함할 수 있고 함수형 인터페이스에서 해당 메서드들은 몇 개를 추가해도 상관이 없습니다.
함수형 인터페이스 생성 시 @FunctionalInterface
어노테이션을 사용하면 해당 인터페이스가 함수형 인터페이스가 맞는지 검사합니다. 따라서 인터페이스 검증과 가독성을 위해 함수형 인터페이스 생성 시 @FunctionalInterface
어노테이션 사용을 권장합니다.
2. 람다 표현식(Lambda Expressions)
함수형 인터페이스를 사용하면 Java8 부터 추가된 람다 표현식(Lambda Expressions)으로 구현체를 생성하여 사용할 수 있습니다. 아래는 그 예시입니다.
public class Foo {
public static void main(String[] args) {
// Java8 이전에는 아래와 같이 구현해서 사용했어야 합니다. - 익명 내부 클래스(anonymous inner class)
RunSomething runSomething = new RunSomething() {
@Override
public void doIt() {
System.out.println("Hello");
}
};
// Java8 이후
RunSomething runSomethingLambda = () -> System.out.println("Hello");
runSomethingLambda.doIt(); // -> Hello
}
}
3. Java 에서 기본으로 제공하는 함수형 인터페이스
Java 에서는 기본적으로 제공하는 함수형 인터페이스가 있습니다. 그 중 자주 사용하는 인터페이스는 아래와 같습니다.
Interface
Description
Function<T, R>
Input : T, Output : R
BiFunction<T, U, R>
입력 값을 2개 받습니다.
Consumer<T>
Return 값이 없고 받기만 합니다.
Supplier<T>
입력 값이 없고 return 값만 있습니다.
Predicate<T>
입력 값을 받고 boolean 값을 반환합니다.
UnaryOperator<T>
입력 값의 타입과 반환 값의 타입이 같은 경우에 사용합니다.
BinaryOperator<T>
BiFunctional 의 2개의 입력 값과 1개의 반환 값의 타입이 같은 경우에 사용합니다.
아래는 코드 예시입니다.
import java.util.function.*;
public class Foo {
public static void main(String[] args) {
// Function<T, R>
Function<Integer, Integer> plus10 = (i) -> i + 10;
Function<Integer, Integer> multiply2 = (i) -> i * 2;
// compse 와 andThen 으로 조합도 가능합니다.
System.out.println(multiply2.compose(plus10).apply(2));
System.out.println("andThen.apply(2) = " + plus10.andThen(multiply2).apply(2));
// BiFunction<T, U, R>
BiFunction<Integer, String, String> printAge = (i, s) -> i + s;
System.out.println("printAge.apply(3, 살) = " + printAge.apply(3, "살"));
// Consumer<T>
Consumer<Integer> printT = (i) -> System.out.println(i);
printT.accept(10);
// Supplier<T>
Supplier<Integer> get10 = () -> 10;
System.out.println("get10.get() = " + get10.get());
// Predicate<T>
Predicate<String> startWithDev = (s) -> s.startsWith("dev");
System.out.println(startWithDev.test("devlee"));
// negate() 는 반대의 결과 값을 반환합니다. or(), and() 연산자도 있습니다.
System.out.println(startWithDev.negate().test("devlee"));
// UnaryOperator<T>, Function<T, R> 인터페이스를 상속 받습니다.
UnaryOperator<Integer> plus10Unary = (i) -> i + 10;
System.out.println(plus10Unary.apply(2));
// BinaryOperator<T>
BinaryOperator<Integer> multiply = (i, j) -> i * j;
System.out.println(multiply.apply(2, 8));
}
}
그 외 인터페이스 : Java Docs
Last updated