열거한 값들이 주로 단독이 아닌 집합으로 사용될 경우, 예전에는 각 상수에 서로 다른 2의 거듭제곱 값을 할당한 정수 열거 패턴을 사용해 왔다.
비트 필드
비트 필드 열거 상수
아래처럼 비트별 OR를 사용해 여러 상수를 하나의 집합으로 모을 수 있으며, 이렇게 만들어진 집합을 비트 필드라 한다.
text.applyStyles(STYLE_BOLD | STYLE_ITALIC);
- 비트 필드를 사용하면 비트별 연산을 사용해 합집합과 교집합 같은 집합 연산을 효율적으로 수행할 수 있다.
- 비트 필드는 정수 열거 상수의 단점을 그대로 지닌다.
- 비트 필드 값이 그대로 출력되면 단순한 정수 열거 상수를 출력할 때보다 해석하기 훨씬 어렵다.
- 비트 필드 하나에 녹아 있는 모든 원소를 순회하기도 까다롭다.
- 최대 몇 비트가 필요한지 API 작성 시 미리 예측하여 적절한 타입을 선택해야 한다.
EnumSet
java.util 패키지의 EnumSet 클래스는 열거 타입 상수의 값으로 구성된 집합을 효과적으로 표현해 준다.
- Set 인터페이스를 완벽히 구현했다.
- 타입 안전하다.
- 다른 어떤 Set 구현체와도 함께 사용하라 수 있다.
- 내부는 비트 벡터로 구현되어 우수한 성능을 가진다. (원소가 총 64개 이하 즉, 대부분의 경우에 EnumSet 전체를 long 변수 하나로 표현하여 비트 필드에 비견되는 성능을 보여준다.)
- 비트를 직접 다룰 때 흔히 겪는 오류들에서 해방된다. EnumSet이 다 처리해 준다.
- removeAll과 retainAll 같은 대량 작업은 비트를 효율적으로 처리할 수 있는 산술 연산을 써서 구현했다.
EnumSet을 사용한 Text 클래스
main은 applyStyles 메서드에 EnumSet 인스턴스를 건네는 클라이언트 코드이다.
EnumSet은 집합 생성 등 다양한 기능의 정적 팩터리를 제공하는데 여기서는 그중 of 메서드를 사용했다.
EnumSet of 메서드
EnumSet의 of 메서드는 지정된 열거형 상수들로 구성된 새로운 EnumSet을 생성하는 역할을 한다.
applyStyles 메서드가 EnumSet <Style>이 아닌 Set <Style>을 받을까?
- 모든 클라이언트가 EnumSet을 건네리라 짐작되는 상황이라도 이왕이면 인터페이스로 받는 게 좋은 습관이다.
- 특이한 클라이언트가 다른 Set 구현체를 넘기더라도 처리할 수 있다.
결론
- 열거할 수 있는 타입을 한데 모아 집합 형태로 사용한다고 해도 비트 필드를 사용할 이유는 없다.
- EnumSet을 사용하자.
'Java > Effective Java' 카테고리의 다른 글
[Effective Java] Item 38. 확장할 수 있는 열거 타입이 필요하면 인터페이스를 사용하라 (0) | 2023.06.29 |
---|---|
[Effective Java] Item 37. ordinal 인덱싱 대신 EnumMap을 사용하라 (0) | 2023.06.29 |
[Effective Java] Item 35. ordinal 메서드 대신 인스턴스 필드를 사용하라 (0) | 2023.05.16 |
[Effective Java] Item 34. int 상수 대신 열거 타입을 사용하라 (0) | 2023.05.16 |
[Effective Java] Item 33. 타입 안전 이종 컨테이너를 고려하라 (0) | 2023.05.16 |