-
아이템36. 비트 필드 대신 EnumSet을 사용하라JAVA/Effective java 2021. 3. 26. 11:10
- 열거한 값들이 주로 단독이 아닌 집합으로 사용될 경우, 예전에는 각 상수에 서로 다른 2의 거듭제곱 값을 할당한 정수 열거 패턴을 사용해왔다.
- 하지만 비트 필드는 정수 열거 상수의 단점을 그대로 지니며, 추가로 다음과 같은 문제까지 안고 있다.
- 비트 필드 값이 그대로 출력되면 단순한 정수 열거 상수를 출력할 때보다 해석하기가 훨씬 어렵다.
- 비트 필드 하나에 녹아 있는 모든 원소를 순회하기도 까다롭다.
- 마지막으로 최대 몇 비트가 필요한지를 API 작성 시 미리 예측하여 적절한 타입을 선택해야 한다.
- 이보다 더 나은 대안이 있는데 java.util 패키지의 EnumSet 클래스는 열거 타입 상수의 값으로 구성된 집합을 효과적으로 표현해준다.
- Set 인터페이스를 완벽히 구현하며, 타입 안전하고, 다른 어떤 Set 구현체와도 함께 사용할 수 있다.
- 하지만 EnumSet의 내부는 비트 벡터로 구현되었다. 비트를 효율적으로 처리할 수 있는 산술 연산을 써서 구현했지만 비트를 직접 다룰 때 흔히 겪는 오류들에서 해방된다.
- 난해한 작업을 EnumSet이 다 처리해주기 때문이다.
- 앞의 예를 열거 타입과 EnumSet을 사용해 수정해보자. 보다시피 짧고 깔끔하고 안전하다.
- 다음은 applyStyles 메서드에 EnumSet 인스턴스를 건네는 클라이언트 코드다.
- EnumSet은 집합 생성 등 다양한 기능의 정적 팩터리를 제공하는데, 다음 코드에서는 그중 of 메서드를 사용했다.
- applyStyles 메서드가 EnumSet<Style>이 아닌 Set<Style>을 받은 이유를 생각해보자.
- 모든 클라이언트가 EnumSet을 건네리라 짐작되는 상황이라도 이왕이면 인터페이스로 받는 게 일반적으로 좋은 습관이다.
- 이렇게 하면 좀 특이한 클라이언트가 다른 Set 구현체를 넘기더라도 처리할 수 있으니 말이다.
핵심 정리
- 열거할 수 있는 타입을 한데 모아 집합 형태로 사용한다고 해도 비트 필드를 사용할 이유는 없다.
- EnumSet 클래스가 비트 필드 수준의 명료함과 성능을 제공하기 때문이다.
- EnumSet의 유일한 단점이라면 불변 EnumSet을 만들 수 없다는 것이다. 이는 Collections..unmodifiableSet으로 EnumSet을 감싸 사용할 수 있다.
참고 자료
'JAVA > Effective java' 카테고리의 다른 글
아이템 38. 확장할 수 있는 열거 타입이 필요하면 인터페이스를 사용하라 (0) 2021.04.14 아이템 37. ordinal 인덱싱 대신 EnumMap을 사용하라. (0) 2021.03.28 아이템 35. ordinal 메서드 대신 인스턴스 필드를 사용하라 (0) 2021.03.26 아이템34. int 상수 대신 열거 타입을 사용하라 (0) 2021.03.23 아이템 33. 타입 안전 이종 컨테이너를 고려하라 (0) 2021.03.22