개발하는 두더지

[Effective Java 규칙32] 비트 필드(bit field) 대신 EnumSet을 사용해라 본문

Java,Android

[Effective Java 규칙32] 비트 필드(bit field) 대신 EnumSet을 사용해라

덜지 2018. 10. 12. 11:13

[Effective Java 규칙32] 비트 필드(bit field) 대신 EnumSet을 사용해라

Effective Java 2/E 책과 구글링을 통해 내용을 정리하고 개인적인 견해가 포함된 글입니다.


EnumSet은 비트 필드만큼 간결하고 성능이 우수합니다. 

아래와 같은 클래스를 보면,

class Text {
public static final int STYLE_BOLD = 1 << 0;
public static final int STYLE_ITALIC = 1 << 1;
public static final int STYLE_UNDERLINE = 1 << 2;
public static final int STYLE_STRIKETHROUGH = 1 << 3;

public void applyStyles(int styles) { // ... }
}

스타일을 적용할 때 상수를 or 연산을 아용하여 아래와 같이 클라이언트에서 사용할 수 있습니다.

text.applyStyles(STYLE_BOLD | STYLE_ITALIC);


일반적으로 list, set보다 bit 연산으로 처리하면 속도가 훨씬 빠르다는 장점이 있지만 위의 코드의 경우에는 단점이 있습니다.

연산한 값으로 어떤 값들로 조합이 이루어졌는지 알 수 없어서 다시 비트 연산을 통해서 어떤 상수로 조합이되었는지 찾아야합니다. 그리고 상수에 해당되는 값이 무엇을 의미하는지 직관적으로 알아보기 어렵습니다. STYLE_ 과 같은 prefix가 없다면 더더욱 보기 어렵습니다.


그래서 이런 단점을 없애기위해 Set의 구현체인  EnumSet가 생겼습니다. 내부적으로 비트 백터(bit vector)를 사용하고, enum 값 개수가 64개 이하인 경우에는 EnumSet은 long 값 하나만 사용합니다. 그래서 본문 맨처음에도 설명했듯이 비트 필드만큼의 성능이 나오게되는 것입니다. 


EnumSet을 이용하여 비트 필드의 단점을 없앤 코드입니다.

class Text {
enum Style { BOLD, ITALIC, UNDERLINE, STRIKETHROUGH }

void applyStyles(Set<Style> styles) { }

public static void main(String[] args) {
Text text = new Text();
text.applyStyles(EnumSet.of(Style.BOLD, Style.ITALIC));
}
}














Comments