ordinal을 인덱스의 배열로 사용한 경우
• 제네릭과 호환되지 않아 비검사 형변환을 수행해야 한다. • 배열 인덱스 (ordinal) 만으로는 의미를 알 수 없으니 출력할 때 레이블을 달아 야 한다. • 정확한 정수값 (ordinal)을 사용할 것을 개발자가 보장해야 한다. (타입 안정성 이 보장되지 않음)
열거 타입(enum)을 키로 사용하도록 설계한 아주 빠른 Map 구현체, EnumMap 을 사용하면 모든 문제를 해결할 수 있다.
// EnumMap을 사용해 열거 타입에 데이터를 연관시키기 (226-228쪽)
// 식물을 아주 단순하게 표현한 클래스 (226쪽)
class Plant {
enum LifeCycle { ANNUAL, PERENNIAL, BIENNIAL }
final String name;
final LifeCycle lifeCycle;
Plant(String name, LifeCycle lifeCycle) {
this.name = name;
this.lifeCycle = lifeCycle;
}
@Override public String toString() {
return name;
}
public static void main(String[] args) {
Plant[] garden = {
new Plant("바질", LifeCycle.ANNUAL),
new Plant("캐러웨이", LifeCycle.BIENNIAL),
new Plant("딜", LifeCycle.ANNUAL),
new Plant("라벤더", LifeCycle.PERENNIAL),
new Plant("파슬리", LifeCycle.BIENNIAL),
new Plant("로즈마리", LifeCycle.PERENNIAL)
};
// 코드 37-2 EnumMap을 사용해 데이터와 열거 타입을 매핑한다. (227쪽)
**Map<LifeCycle, Set<Plant>> plantsByLifeCycle =
new EnumMap<>(LifeCycle.class);**
for (LifeCycle lc : LifeCycle.values())
plantsByLifeCycle.put(lc, new HashSet<>());
for (Plant p : garden)
plantsByLifeCycle.get(p.lifeCycle).add(p);
System.out.println(plantsByLifeCycle);
// 코드 37-3 스트림을 사용한 코드 1 - EnumMap을 사용하지 않는다! (228쪽)
Map<LifeCycle, List<Plant>> collect = Arrays.stream(garden)
.collect(groupingBy(p -> p.lifeCycle));
System.out.println(collect);
// 코드 37-4 스트림을 사용한 코드 2 - EnumMap을 이용해 데이터와 열거 타입을 매핑했다. (228쪽)
EnumMap<LifeCycle, Set<Plant>> collect1 = Arrays.stream(garden)
.collect(groupingBy(p -> p.lifeCycle,
() -> new EnumMap<>(LifeCycle.class), toSet()));
System.out.println(collect1);
}
}
// 코드 37-6 중첩 EnumMap으로 데이터와 열거 타입 쌍을 연결했다. (229-231쪽)
public enum Phase {
SOLID, LIQUID, GAS;
public enum Transition {
MELT(SOLID, LIQUID), FREEZE(LIQUID, SOLID),
BOIL(LIQUID, GAS), CONDENSE(GAS, LIQUID),
SUBLIME(SOLID, GAS), DEPOSIT(GAS, SOLID);
// // 코드 37-7 EnumMap 버전에 새로운 상태 추가하기 (231쪽)
// SOLID, LIQUID, GAS, PLASMA;
// public enum Transition {
// MELT(SOLID, LIQUID), FREEZE(LIQUID, SOLID),
// BOIL(LIQUID, GAS), CONDENSE(GAS, LIQUID),
// SUBLIME(SOLID, GAS), DEPOSIT(GAS, SOLID),
// IONIZE(GAS, PLASMA), DEIONIZE(PLASMA, GAS);
private final Phase from;
private final Phase to;
Transition(Phase from, Phase to) {
this.from = from;
this.to = to;
}
**// 해당 코드 정도가 적당하다.**
private static final Map<Phase, Map<Phase, Transition>> m = new EnumMap<>(Phase.class);
static {
for (Transition t : Transition.values()) {
m.computeIfAbsent(t.from, k -> new EnumMap<>(Phase.class)).put(t.to, t);
}
}
// static {
// for (Transition t : Transition.values()) {
// Map<Phase, Transition> innerMap = m.get(t.from);
// if (innerMap == null) {
// innerMap = new EnumMap<>(Phase.class);
// m.put(t.from, innerMap);
// }
// innerMap.put(t.to, t);
// }
// }
// @formatter:off
// 상전이 맵을 초기화한다.
private static final Map<Phase, Map<Phase, Transition>>
m = Stream.of(values()).collect(groupingBy(
t -> t.from,
() -> new EnumMap<>(Phase.class),
toMap(t -> t.to, t -> t, (x, y) -> y, () -> new EnumMap<>(Phase.class)))
);
// @formatter:on
public static Transition from(Phase from, Phase to) {
return m.get(from).get(to);
}
}
// 간단한 데모 프로그램 - 깔끔하지 못한 표를 출력한다.
public static void main(String[] args) {
for (Phase src : Phase.values()) {
for (Phase dst : Phase.values()) {
Transition transition = Transition.from(src, dst);
if (transition != null)
System.out.printf("%s에서 %s로 : %s %n", src, dst, transition);
}
}
}
}
Enum을 Key로 사용하는 Map
EnumMap은 내부 자료 구조로 배열을 사용한다. HashMap에 비해 빠르다.
public class EnumMapExample {
enum LifeCycle {
ANNUAL, PERENNIAL, BIENNIAL
}
public static void main(String[] args) {
Map<LifeCycle, String> plantsByLifeCycle = new EnumMap<>(LifeCycle.class);
plantsByLifeCycle.put(LifeCycle.PERENNIAL, "라벤더");
plantsByLifeCycle.put(LifeCycle.ANNUAL, "바질");
plantsByLifeCycle.put(LifeCycle.BIENNIAL, "파슬리");
System.out.println(plantsByLifeCycle);
// 동시성 문제를 해결하기 위한 MAP
Map<LifeCycle, String> plantsByLifeCycle2 = Collections.synchronizedMap(new EnumMap<>(LifeCycle.class));
plantsByLifeCycle2.put(LifeCycle.PERENNIAL, "라벤더");
}
}
Enum이 정의된 순서를 따르는 컬렉션이다.
null값을 key로 허용하지 않는다.
동기화 되어 있지 않다.
https://docs.oracle.com/en%2Fjava%2Fjavase%2F11%2Fdocs%2Fapi%2F%2F/java.base/java/ util/EnumMap.html