列挙セットを使用して、単一のフィールドに2ビットを組み合わせるにはどうすればよいですか?
-
27-10-2019 - |
質問
1バイトでコーディングされたバイナリ文字列を考えると、そのバイトのビットのいくつかを列挙値にマッピングすることは可能ですか?
例:私がこれに自分のフィールドを切り取りたいとしましょう:
- bit1およびbit2 = field1
- bit3 = field2
- bit4 = field3
- bit5 = field4
- 他のビットは未使用です
どうすればそれをマッピングできますか?
public enum MyEnum {
FIELD1, FIELD2, FIELD3, FIELD4;
private static final EnumSet<MyEnum> ALLFIELDS = EnumSet.allOf(MyEnum.class);
}
私がやろうとしているのは、ビットマスクを使用してこれらの列挙値を除外することです。そのために、私は次の方法を行いました:
public static <E extends Enum<E>> EnumSet<E> filterWithBitMask(EnumSet<E> set, int bitmask) {
List<E> kept = new ArrayList<E>();
for (E property : set) {
if ((bitmask & (1 << ((Enum<E>) property).ordinal())) != 0) {
kept.add(property);
}
}
EnumSet<E> selectedProperties = set.clone();
selectedProperties.retainAll(kept);
return selectedProperties;
}
これは、列挙フィールドが1つのビットを表す限り、正常に機能しますが、それらを単一のフィールドに組み合わせる方法を理解することはできません。申し訳ありませんが、それが明白に見える場合は申し訳ありませんが、ビットレベルでデータを操作するのはこれが初めてです...
コメントの編集:私は実際に持っているものを表すためにフィールド構造を編集しました。したがって、1ビット以上にわたる唯一のフィールドはfield1です。 BIT1とBIT2の存在に基づいて、Field1に重み変数を割り当てます。
public enum MyByte {
BIT_1, BIT_2, BIT_3, BIT_4, BIT_5;
private static final EnumSet<MyByte> ALLPROPERTIES = EnumSet.allOf(MyByte.class);
public static EnumSet<Fields> getValues(int bitmask) {
EnumSet<MyByte> selectedBits = filterWithBitMask(ALLPROPERTIES, bitmask);
int weight = 0;
EnumSet<Fields> fields = null;
if (selectedBits.contains(BIT_1))
weight += 1;
if (selectedBits.contains(BIT_2))
weight += 2;
if (selectedBits.contains(BIT_1) || selectedBits.contains(BIT_2))
fields = EnumSet.of(Fields.FIELD1.setValue(weight));
if (selectedBits.contains(BIT_3)) {
if (fields != null)
fields.add(Fields.FIELD2);
else fields = EnumSet.of(Fields.FIELD2);
}
if (selectedBits.contains(BIT_4)) {
if (fields != null)
fields.add(Fields.FIELD3);
else fields = EnumSet.of(Fields.FIELD3);
}
if (selectedBits.contains(BIT_5)) {
if (fields != null)
fields.add(Fields.FIELD4);
else fields = EnumSet.of(Fields.FIELD4);
}
return fields;
}
public enum Fields {
// BIT_1 and BIT_2
FIELD1,
// BIT_3
FIELD2,
// BIT_4
FIELD3,
// BIT_5
FIELD4;
private int value;
public Fields setValue(int i) {
this.value = i;
return this;
}
}
}
これが私が今から思いついた汚い解決策ですが、私はそれらのすべての声明でそれを本当に好きではありません。誰かがより良い解決策を持っているなら、私はこれを変えることができてうれしいです;-)
ありがとう !
解決
あなたが何を望んでいるのかよくわかりません。これは興味深いかもしれません:
import java.util.*;
enum Field {
field1((byte) 0xc0) {
int filter_(byte mask) { // you may want this one to be different
return mask & this.mask;
}
},
field2((byte) 0x20), field3((byte) 0x10), field4((byte) 0x08);
Field(byte mask) {
this.mask = mask;
}
int filter_(byte mask) {
return mask & this.mask;
}
static EnumSet<Field> filter(byte mask) {
final EnumSet<Field> fields = EnumSet.noneOf(Field.class);
for (Field field : values())
if (field.filter_(mask) != 0) fields.add(field);
return fields;
}
final byte mask;
}
public class Main {
public static void main(String[] args) {
for(Field field:Field.values()) {
System.out.println(Field.filter(field.mask));
}
}
}
所属していません StackOverflow