#java #postgresql #hibernate #jpa
#java #postgresql #спящий режим #jpa
Вопрос:
У меня есть таблица со столбцом, тип которого равен bit (24). Я пытался использовать String, и когда я пытаюсь получить этот объект, он хорошо сопоставляется, но когда я пытаюсь его сохранить, Hibernate выдает исключение: org.postgresql.util.PSQLException: ERROR: column is of type bit but expression is of type character varying
Я знаю, что если это бит (1), я могу использовать логическое значение, но я не знаю, что мне делать в моем случае.
Заранее спасибо!
Ответ №1:
К сожалению, JPA / hibernate не поддерживает БИТ, имеющий параметризованную длину.
К счастью, вы можете определить свой собственный пользовательский тип с помощью hibernate.
Сначала вам нужно создать BitStringType
, BitStringJavaDescriptor
, и BitStringSqlDescriptor
:
public class BitStringType extends AbstractSingleColumnStandardBasicType<String> {
public static final BitStringType INSTANCE = new BitStringType();
public BitStringType() {
super(VarcharTypeDescriptor.INSTANCE, BitStringJavaDescriptor.INSTANCE);
}
@Override
public String getName() {
return "BitString";
}
}
public class BitStringJavaDescriptor extends AbstractTypeDescriptor<String> {
public static final BitStringJavaDescriptor INSTANCE = new BitStringJavaDescriptor();
public BitStringJavaDescriptor() {
super(String.class, ImmutableMutabilityPlan.INSTANCE);
}
@Override
public String fromString(String string) {
return string;
}
@Override
public <X> X unwrap(String value, Class<X> type, WrapperOptions options) {
if (value == null)
return null;
if (String.class.isAssignableFrom(type))
return (X) value;
throw unknownUnwrap(type);
}
@Override
public <X> String wrap(X value, WrapperOptions options) {
if (value == null)
return null;
if (String.class.isInstance(value))
return (String) value;
throw unknownWrap(value.getClass());
}
}
public class BitStringSqlDescriptor implements SqlTypeDescriptor {
public static BitStringSqlDescriptor INSTANCE = new BitStringSqlDescriptor();
@Override
public int getSqlType() {
return Types.OTHER;
}
@Override
public boolean canBeRemapped() {
return true;
}
@Override
public <X> ValueBinder<X> getBinder(final JavaTypeDescriptor<X> javaTypeDescriptor) {
return new BasicBinder<X>(javaTypeDescriptor, this) {
@Override
protected void doBind(PreparedStatement st, X value, int index, WrapperOptions options) throws SQLException {
st.setObject(index, javaTypeDescriptor.unwrap(value, String.class, options), Types.OTHER);
}
@Override
protected void doBind(CallableStatement st, X value, String name, WrapperOptions options) throws SQLException {
st.setObject(name, javaTypeDescriptor.unwrap(value, String.class, options), Types.OTHER);
}
};
}
@Override
public <X> ValueExtractor<X> getExtractor(final JavaTypeDescriptor<X> javaTypeDescriptor) {
return new BasicExtractor<X>(javaTypeDescriptor, this) {
@Override
protected X doExtract(ResultSet rs, String name, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(rs.getString(name), options);
}
@Override
protected X doExtract(CallableStatement statement, int index, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(statement.getString(index), options);
}
@Override
protected X doExtract(CallableStatement statement, String name, WrapperOptions options) throws SQLException {
return javaTypeDescriptor.wrap(statement.getString(name), options);
}
};
}
}
Имея эти классы, вы можете определить тип для своего поля. Пожалуйста, используйте правильный пакет (в моем случае я использовал тот, который был в моей демонстрации com.yonlabs.jpa
):
@Column
@Type(type = "com.yonlabs.jpa.BitStringType")
private String bits;
Вы также можете зарегистрировать этот тип в hibernate, чтобы использовать зарегистрированное имя вместо полного класса Java.