코딩못하는사람

아이템43] 람다보다는 메서드 참조를 사용하라 본문

자바 메모장/이펙티브 자바

아이템43] 람다보다는 메서드 참조를 사용하라

공부절대안함 2021. 8. 16. 19:33

아이템43] 람다보다는 메서드 참조를 사용하라

람다의 장점은 익명클래스보다 코드가 간결해진다는 장점을 가진다.

하지만 여기서 메서드 참조(Method reference)를 활용하면 한단계 더 간결해진 코드를 만들 수 있다.

 

코드로 예시를 보자.

Map에 키가 없다면 키와 숫자 1을 매핑하고, 키가 존재하면 기존 매핑값을 증가시키는 코드이다.

map.merge(key, 1, (count, incr) -> count + incr);
//Map에서 제공하는 merge 메서드

merge 메서드는 키,값,함수를 인자로 받는다. 깔끔해 보이지만 매개변수 count와 incr이 하는일 없이 코드를 차지한다.

자바 8에서는 Integer 클래스 및 기본타입 래퍼클래스은 이 람다와 기능이 같은 정적 메서드 sum 을 제공한다.

 

따라서 다음과 같이 람다대신 메서드 참조를 사용하면 코드가 더 간결해진다.

map.merge(key, 1, Integer::sum);

매개변수가 늘어날수록 메서드 참조로 제거할 수 있는 코드양도 늘어난다.

 

하지만 람다의 매개변수 이름 자체가 프로그래머들이 코드를 이해하는데 좋은 가이드가 되기도 한다.

이런 람다는 길이는 길지만 메서드 참조보다 코드 읽기가 좋아서 유지보수에 용이할 수 있다.

 

IDE에서는 메서드 참조를 항상 권하지만 항상 좋은것은 아니다.

다음의 코드가 GoshThisClassNameIsHumongous라는 클래스 안에 있다고 가정해보자

service.execute(GoshThisClassNameIsHumongous::action);
//메서드 참조 

service.execute(() -> acation());
//람다 사용

다음과 같은 경우에는 메서드 참조가 간결하지도 않고 명확하지도 않다.

또 java.util.function패키지의 Function.identity()를 사용하는 것보다 람다(x->x)를 사용하는 것이 코드도 짧고 명확한 경우가 있다.

 

메서드 참조 유형

메서드 참조에는 5가지 유형이 있다.

정적 Integer::parseInt str -> Integer.parseInt(str)
한정적(인스턴스) Instance.now()::isAfter Instance then = Instant.now(); t -> then.isAfter(t)
비한정적(인스턴스) String::toLowerCase str -> str.toLowerCase()
클래스 생성자 TreeMap<K, V>::new () -> new TreeMap<K, V>()
배열 생성자 int[]::new len -> new int[len]

정적: 가장 흔하다, 정적메서드를 가리키는 메서드 참조이다.

 

한정적(인스턴스): 수신객체를 특정하는 한정적(bound) 인스턴스 메서드 참조이다.(정적 참조와 비슷)

  • 함수객체가 받는 인수와 메서드가 받는 인수가 똑같다.

 

비한정적(인스턴스): 수신객체를 특정하지 않는 비한정적(unbound) 인스턴스 메서드.

  • 참조함수객체를 적용하는 시점에 수신객체를 알려준다.
  • 수신객체 전달용 매개변수가 매개변수 목록의 첫번째로 추가되며,그 뒤로는 참조되는 메서드 선언에 정의된 매개변수들이 따라옴.
  • 주로 스트림 파이프라인에서 매핑과 필터함수에 쓰인다.

클래스 및 배열 생성자:생성자를 가리키는 메서드 참조와 배열 생성자를 가리키는 메서드 참조. 생성자 참조는 팩터리 객체로 사용된다.

 

정리:

메서드 참조는 람다의 대안이 될 수 있다. 메서드 참고 쪽이 짧고 명확하면 메서드 참조를 쓰고 아니면 람다를 쓰자.

 

 

Comments