将java BitSet保存到DB

时间:2012-09-12 14:07:38

标签: java jpa persistence bitset

使用JPA,我希望能够将BitSet保存到数据库中并将其拉回去。

假设我有:

@Entity
@Table(name = "myTable")
public class MyClass {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "Object_Id")
    protected long id;

    @Column(name = "Tags")
    protected BitSet tags;

... getters & setters etc...
}

还应该定义“columnDefinition”吗?我真的不明白它是如何持久化的(使用toString()?)以及它如何从数据库中加载回来。

你能帮我解决这个问题吗?

谢谢!

2 个答案:

答案 0 :(得分:5)

更有效的方式(使用int代替byte[])需要一个非常简单的自定义类:

@Entity
@Access(AccessType.FIELD)
public class SampleEntity {

    @Transient
    private IntBitSet isolationLevel = new IntBitSet(0);

    public static final int USER_BIT = 0;
    public static final int DEVICE_BIT = 1;
    // 2, 3, 4, ...

    public boolean isUserIsolated() {
        return isolationLevel.bitGet(USER_BIT);
    }

    public boolean isDeviceIsolated() {
        return isolationLevel.bitGet(DEVICE_BIT);
    }

    public void setUserIsolated(boolean b) {
        isolationLevel.bitSet(USER_BIT, b);
    }

    public void setDeviceIsolated(boolean b) {
        isolationLevel.bitSet(DEVICE_BIT, b);
    }

    @Access(AccessType.PROPERTY)
    @Column
    public int getIsolationLevel() {
        return isolationLevel.getValue();
    }

    public void setIsolationLevel(int isolationLevel) {
        this.isolationLevel = new IntBitSet(isolationLevel);
    }

    private static class IntBitSet {
        private int value;

        public IntBitSet(int value) {
            this.value = value;
        }

        public int getValue() {
            return value;
        }

        public boolean bitGet(int i) {
            return ((value >> i) & 1) == 1;
        }

        public void bitSet(int i, boolean b) {
            if (b) {
                bitSet(i);
            } else {
                bitUnset(i);
            }
        }
        private void bitSet(int i) {
            value = value | (1 << i);
        }
        private void bitUnset(int i) {
            value = value & ~(1 << i);
        }
    }
}

答案 1 :(得分:3)

默认情况下,JPA使用Java序列化来保存未知Serializable类型的属性(以便将序列化表示存储为byte[])。

通常它不是您想要的,因为可以有更有效的方式来表示您的数据。例如,BitSet可以有效地表示为数字(如果其大小有界),或byte[]或其他内容(遗憾的是,BitSet不提供执行这些转换的方法,因此你需要手动实现它们。

当您确定要在数据库中拥有哪些数据表示时,您需要告知JPA应用必要的转换。有两种选择:

  • 在getter和setter中实现转换。例如,如下:

    @Entity
    @Table(name = "myTable")
    @Access(AccessType.FIELD)
    public class MyClass {
        ...
        @Transient // Do not store this field
        protected BitSet tags;
    
        @Access(AccessType.PROPERTY) // Store the property instead
        @Column(name = "Tags")
        byte[] getTagsInDbRepresentation() {
            ... // Do conversion
        }
    
        void setTagsInDbRepresentation(byte[] data) {
            ... // Do conversion
        }
        ...
    }
    
  • 使用特定于提供程序的扩展来隐式执行转换(例如,Hibernate中的custom types)。此方法允许您在不同实体中重用类型转换逻辑。