使用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()?)以及它如何从数据库中加载回来。
你能帮我解决这个问题吗?
谢谢!
答案 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)。此方法允许您在不同实体中重用类型转换逻辑。