본문 바로가기

Reading Record/이펙티브자바

[아이템 36] 비트 필드 대신 EnumSet을 사용하라

열거한 값들이 주로 집합으로 사용될 경우, 예전에는 상수에 서로 다른 2의 거듭제곱 값을 할당한 정수 열거 패턴을 사용해왔다.

package Chap5_EnumTypeAndAnnotation.item36;

public class StyleWithBitField {
    public static final int STYLE_BOLD = 1 << 0; // 1
    public static final int STYLE_ITALIC = 1 << 1; // 2
    public static final int STYLE_UNDERLINE = 1 << 2; // 4
    public static final int STYLE_STRIKETHROUGH = 1 << 3; // 8

    public void applyStyles(int styles) {
//        styleWithBitField.applyStyles(STYLE_BOLD | STYLE_UNDERLINE);
    }
}

위와 같이 비트별 OR를 사용해 여러 상수를 하나의 집합으로 모을 수 있으며, 이 집합을 비트 필드라 한다.

하지만 비트 필드 또한 정수 열거 상수이므로 정수 열거 상수의 단점을 그대로 지니며, 추가로 다른 문제도 가지고있다.

1. 비트 필드 값이 그대로 출력되면 단순한 정수 열거 상수를 출력할 때보다 해석하기가 훨씬 어렵다.

2. 비티 필드 하나에 녹아 있는 모든 원소를 순회하기도 까다롭다.

3. 최대 몇 비트가 필요한지를 처음부터 예상하고 적절한 타입을 선택해야 한다.

다행히도 이젠 더 나은 방안인 java.util 패키지의 EnumSet 클래스가 있다. EnumSet은 열거 타입 상수의 값으로 구성된 집합을 효과적으로 표현해준다. 

1. EnumSet 클래스는 Set인터페이스를 구현하여 어떤 Set 구현체와도 함께 사용할 수 있으며 타입 안전하다.

2. EnumSet 내부는 비트 벡터로 구현되어있으며, 원소가 총 64개 이하라면 EnumSet 전체를 long 변수 하나로 표현하여 비트필드에 비견되는 성능을 보여준다.

3. removeAll과 retainAll 과 같은 대량 작업은 비트를 효율적으로 처리할 수 있는 산술 연산을 써서 구현하였다.

위의 코드를 EnumSet을 이용하여 수정해보자.

import java.util.EnumSet;
import java.util.Set;

public class Text {

    public void applyStyles(Set<Style> styles) {
//        applyStyles(EnumSet.of(Style.BOLD, Style.UNDERLINE));
    }

    private enum Style {
        BOLD, ITALIC, UNDERLINE, STRIKETHROUGH
    }
}

 

 

EnumSet을 사용해서 비트 필드의 단점을 개선하고 열거 상수 집합을 사용할 수 있게되었다. 하지만 그런 EnumSet에도 단점이 존재하는데, 불변 EnumSet을 만들 수 없다는 점이다.

성능상에서 손해를 보게되지만 Collections.unmodifiableSet() 메소드를 사용하여  EnumSet을 불변 객체로 만들 수 있다.