我的同事感叹我对代码生成的坚持,但我决心证明这个特定的链可以完成。下面的例子从我的项目中简化(因此未经测试的类型);我很快就会创建一个测试工具。目前使用的是hibernate 4.1.4,hibernate-spatial 1.1.1,hyperjaxb3-ejb-plugin 0.5.6。
我从一个使用Position类型的模式开始,该元素的类型是一个简单的字符串扩展名:
<xsd:simpleType name="wktPoint">
<xsd:restriction base="xsd:string">
</xsd:restriction>
</xsd:simpleType>
我使用bindings.xjb自定义来使用jaxb:javaType将我的xmlType wktPoint映射到JTS Point javaType,因为我想将JTS Point字段映射到postgis几何列:
<jaxb:javaType name="com.vividsolutions.jts.geom.Point" xmlType="wktPoint" parseMethod="test.Reader.readWKTPoint" printMethod="test.Writer.writeWKTPoint" />
稍后在bindings.xjb中,我开始使用annox:annotation自定义Position元素:
<jaxb:bindings node="xsd:complexType[@name='MyType']//xsd:element[@name='Position']">
<annox:annotate target="getter">
<annox:annotate annox:class="javax.persistence.Basic"/>
<annox:annotate annox:class="javax.persistence.Column" name="POSITION" columnDefinition="GEOMETRY"/>
<annox:annotate annox:class="org.hibernate.annotations.Type" type="org.hibernatespatial.GeometryUserType">
<annox:annotate annox:field="parameters">
<annox:annotate annox:class="org.hibernate.annotations.Parameter" name="dialect" value="postgis"/>
</annox:annotate>
</annox:annotate>
</annox:annotate>
</jaxb:bindings>
但是,这会在MyType.java中生成以下getPosition方法:
/**
* Gets the value of the position property.
* @return
* possible object is
* {@link String }
*/
@Transient
@Basic
@Column(columnDefinition = "GEOMETRY", name = "POSITION")
@Type(parameters = {
@Parameter(name = "dialect", value = "postgis")
}, type = "org.hibernatespatial.GeometryUserType")
public Point getPosition() {
return position;
}
因此,当我启动我的应用程序并且jpa / hibernate开始初始化表时,它会完全跳过POSITION列(因为它标记为@Transient)。
如何防止@Transient出现?我似乎需要说服Hyperjaxb,我们将能够直接将JTS Point类型写入数据库(这就是所有@Type的内容,表明我们希望hibernatespatial在读取和写入Position时使用postgis方言)。关于如何做到的任何想法?
答案 0 :(得分:2)
0.5.6 hyperjaxb-ejb-plugin只是没有正确处理绑定定制。我构建了一个0.5.7-SNAPSHOT,它修改了org.jvnet.hyperjaxb3.ejb.strategy.mapping AttributesMapping.getAttributeMapping(),这样我的自定义jaxb绑定就不再被标记为Transient。我给了它以下的后备检查:
if (isFieldOutlineBasic(fieldOutline)) {
...
} else if (isFieldOutlineComplex(fieldOutline)) {
...
} else {
if (fieldOutline.getRawType() instanceof JClass) {
return context.getBasicMapping();
}
}
这涵盖了我的自定义,它总是显示为JDirectClass(Point)。一旦我做了这个改变,postgis表就是用正确的几何类型创建的,现在我可以接受一个WKT Point字符串,让它作为JTS Point解组,然后用JPA作为PostGIS几何体保存它。我做了还需要重建Hibernate Spatial(现在使用1.1.2-SNAPSHOT)来使用Hibernate4。我将这项工作提交给Karel Maesen,我也会将Hyperjaxb更改提交给Aleksei Valikov,希望未来的版本更加强大。