/* * Copyright 2016 higherfrequencytrading.com * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package net.openhft.lang.io.serialization.impl; import net.openhft.lang.io.Bytes; import net.openhft.lang.io.serialization.BytesMarshaller; import net.openhft.lang.model.constraints.NotNull; import net.openhft.lang.model.constraints.Nullable; import java.util.BitSet; import java.util.LinkedHashMap; import java.util.Map; /** * @author peter.lawrey */ public class EnumBytesMarshaller<E extends Enum<E>> extends ImmutableMarshaller<E> implements BytesMarshaller<E> { private static final StringBuilderPool sbp = new StringBuilderPool(); @SuppressWarnings("unchecked") private final E[] interner = (E[]) new Enum[1024]; private final BitSet internerDup = new BitSet(1024); private final Map<String, E> map = new LinkedHashMap<String, E>(64); private final E defaultValue; private final int mask; public EnumBytesMarshaller(@NotNull Class<E> classMarshaled, E defaultValue) { this.defaultValue = defaultValue; mask = interner.length - 1; for (E e : classMarshaled.getEnumConstants()) { map.put(e.name(), e); int idx = hashFor(e.name()); if (!internerDup.get(idx)) { if (interner[idx] != null) { //noinspection UnqualifiedFieldAccess,AssignmentToNull interner[idx] = null; internerDup.set(idx); } else { interner[idx] = e; } } } } @Override public void write(@NotNull Bytes bytes, @Nullable E e) { bytes.writeUTFΔ(e == null ? "" : e.name()); } private int hashFor(@NotNull CharSequence cs) { int h = 0; for (int i = 0, length = cs.length(); i < length; i++) h = 57 * h + cs.charAt(i); h ^= (h >>> 20) ^ (h >>> 12); h ^= (h >>> 7) ^ (h >>> 4); return h & mask; } @Override public E read(@NotNull Bytes bytes) { StringBuilder sb = sbp.acquireStringBuilder(); bytes.readUTFΔ(sb); return builderToEnum(sb); } private E builderToEnum(StringBuilder sb) { int num = hashFor(sb); int idx = num & mask; E e = interner[idx]; if (e != null) return e; if (!internerDup.get(idx)) return defaultValue; e = map.get(sb.toString()); return e == null ? defaultValue : e; } }