48

私は大規模なドキュメント アーカイブを維持しており、ビット フィールドを使用して、処理中または検証中にドキュメントのステータスを記録することがよくあります。私のレガシー コードでは、次のような static int 定数を使用しています。

static int DOCUMENT_STATUS_NO_STATE = 0
static int DOCUMENT_STATUS_OK = 1
static int DOCUMENT_STATUS_NO_TIF_FILE = 2
static int DOCUMENT_STATUS_NO_PDF_FILE = 4

これにより、適切なフラグを設定することで、ドキュメントの状態を簡単に示すことができます。例えば:

status = DOCUMENT_STATUS_NO_TIF_FILE | DOCUMENT_STATUS_NO_PDF_FILE;

静的定数を使用するアプローチは悪い習慣であり、コードを改善したいので、列挙型を使用して同じことを達成しようとしていました。いくつかの要件があります。そのうちの 1 つは、ステータスを数値型としてデータベースに保存する必要があることです。したがって、列挙定数を数値に変換する必要があります。以下は私の最初のアプローチですが、これが正しい方法なのだろうか?

class DocumentStatus{

    public enum StatusFlag {

        DOCUMENT_STATUS_NOT_DEFINED(1<<0),
        DOCUMENT_STATUS_OK(1<<1), 
        DOCUMENT_STATUS_MISSING_TID_DIR(1<<2),
        DOCUMENT_STATUS_MISSING_TIF_FILE(1<<3),
        DOCUMENT_STATUS_MISSING_PDF_FILE(1<<4),
        DOCUMENT_STATUS_MISSING_OCR_FILE(1<<5),
        DOCUMENT_STATUS_PAGE_COUNT_TIF(1<<6),
        DOCUMENT_STATUS_PAGE_COUNT_PDF(1<<7),
        DOCUMENT_STATUS_UNAVAILABLE(1<<8);


        private final long statusFlagValue;

        StatusFlag(long statusFlagValue) {
            this.statusFlagValue = statusFlagValue;
        }

        public long getStatusFlagValue(){
            return statusFlagValue;
        } 

       }


    /**
     * Translates a numeric status code into a Set of StatusFlag enums
     * @param numeric statusValue 
     * @return EnumSet representing a documents status
     */
    public EnumSet<StatusFlag> getStatusFlags(long statusValue) {
        EnumSet statusFlags = EnumSet.noneOf(StatusFlag.class);
        StatusFlag.each { statusFlag -> 
            long flagValue = statusFlag.statusFlagValue
            if ( (flagValue&statusValue ) == flagValue ) {
               statusFlags.add(statusFlag);
            }
        }
        return statusFlags;
    }


    /**
     * Translates a set of StatusFlag enums into a numeric status code
     * @param Set if statusFlags
     * @return numeric representation of the document status 
     */
    public long getStatusValue(Set<StatusFlag> flags) {
        long value=0;
        flags.each { statusFlag -> 
            value|=statusFlag.getStatusFlagValue() 
        }
        return value;
    }

     public static void main(String[] args) {

        DocumentStatus ds = new DocumentStatus();
        Set statusFlags = EnumSet.of(
            StatusFlag.DOCUMENT_STATUS_OK,
            StatusFlag.DOCUMENT_STATUS_UNAVAILABLE);

        assert ds.getStatusValue( statusFlags )==258 // 0000.0001|0000.0010

        long numericStatusCode = 56;
        statusFlags = ds.getStatusFlags(numericStatusCode);

        assert !statusFlags.contains(StatusFlag.DOCUMENT_STATUS_OK);
        assert statusFlags.contains(StatusFlag.DOCUMENT_STATUS_MISSING_TIF_FILE);
        assert statusFlags.contains(StatusFlag.DOCUMENT_STATUS_MISSING_PDF_FILE);
        assert statusFlags.contains(StatusFlag.DOCUMENT_STATUS_MISSING_OCR_FILE);

    }

}
4

5 に答える 5

34

コンストラクターのパラメーターを定義する代わりに、単純に内部ordinal()値を使用してこれを計算できます。

public enum StatusFlag {

    DOCUMENT_STATUS_NOT_DEFINED,
    DOCUMENT_STATUS_OK, 
    DOCUMENT_STATUS_MISSING_TID_DIR,
    DOCUMENT_STATUS_MISSING_TIF_FILE,
    DOCUMENT_STATUS_MISSING_PDF_FILE,
    DOCUMENT_STATUS_MISSING_OCR_FILE,
    DOCUMENT_STATUS_PAGE_COUNT_TIF,
    DOCUMENT_STATUS_PAGE_COUNT_PDF,
    DOCUMENT_STATUS_UNAVAILABLE;


    public long getStatusFlagValue(){
        return 1 << this.ordinal();
    } 

}

エントリの並べ替え、挿入 (最後以外)、または削除を控える必要があることに注意してください。そうしないと、フラグの値が変更され、データベースの内容の意味が変更されます。

于 2011-03-18T01:27:37.093 に答える
20

あなたのアプローチはまさにそれを行う方法です。

于 2011-03-18T00:14:59.567 に答える
6

列挙値を指定しないでください。anEnumSetを使用してそれらを結合し、Enum.ordinal()単一の整数との間で変換するために永続化するときに使用します。Class.getEnumConstants()整数からセットを再構築するときにも役立つ場合があります。

于 2011-03-18T00:07:34.317 に答える
4

この問題のための完全なライブラリを作成しました: http://claude-martin.ch/enumbitset/

主な目標は、列挙型のセットをビットフィールドに格納することでした。ただし、他のタイプもサポートしています。

これにより、「getStatusFlags()」のような追加のメソッドは必要ありません。インターフェイスEnumBitSetHelperを追加するだけで、既存の列挙型で使用できます(「特性」のように使用されます)。各列挙型定数は、Java の EnumSet および BitSet のすべてのメソッドを持つ「EnumBitSet」を作成できます。次に、これらの一連の列挙型定数を操作して、ビットフィールド値に変換できます。

値をビット フィールドに簡単に格納するために、BigInteger や long などの多くの形式をサポートしています。ただし、これは Java バージョン 8 以降でのみ機能することに注意してください。

于 2014-03-13T10:57:04.517 に答える