我编写了一个扩展JAXB注释bean(BaseBeanEx
)的bean(BaseBean
)。 BaseBean
位于数据结构中的某个列表中,无法更改。只要需要,软件会对BaseBeanEx
进行显式转换。我还写了一个ObjectFactory
来创建BaseBeanEx
而不是BaseBean
。这一切都运行正常,但现在我向afterUnmarshal
添加了一个永远不会被调用的BaseBeanEx
方法。
这是一个错误还是根据规格?如果以后是这种情况,是否有一些优雅的工作?
我正在使用默认的JAXB引擎。
答案 0 :(得分:5)
注意:我是EclipseLink JAXB (MOXy)主管,是JAXB 2 (JSR-222)专家组的成员。
afterUnmarshal
未调用BaseBeanEx
的原因是元数据是在BaseBean
类上构建的。为了让您的用例工作,您需要让您的JAXB impl知道您确实要映射到BaseBeanEx
的实例。
选项#1 - 使用注释的任何JAXB实现
的根强> 的
您可以使用@XmlElement
注释覆盖字段/属性的类型。在下面的示例中,方法的签名为List<BaseBean>
,但@XmlElement
注释通知JAXB实现,属性应解释为List<BaseBeanEx>
。
package forum10174513;
import java.util.List;
import javax.xml.bind.annotation.*;
@XmlRootElement
public class Root {
private List<BaseBean> baseBeans;
@XmlElement(name="base-bean", type=BaseBeanEx.class)
public List<BaseBean> getBaseBeans() {
return baseBeans;
}
public void setBaseBeans(List<BaseBean> baseBeans) {
this.baseBeans = baseBeans;
}
}
选项#2 - 使用MOXy的外部映射文档
BaseBean位于数据结构中的某个List中,不可以 改变。
如果您无法修改域模型并使用MOXy作为JAXB提供程序,则可以利用其外部映射文档来应用元数据,而无需修改域模型。
的 bindings.xml 强> 的
<?xml version="1.0"?>
<xml-bindings
xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
package-name="forum10174513">
<java-types>
<java-type name="Root">
<java-attributes>
<xml-element
java-attribute="baseBeans"
name="base-bean"
type="forum10174513.BaseBeanEx"/>
</java-attributes>
</java-type>
</java-types>
</xml-bindings>
的演示强> 的
下面是一些代码,演示如何引导利用外部映射文档的JAXBContext
。目前存在一个错误,即仅通过外部映射文档引用的类将不会注册事件方法(http://bugs.eclipse.org/376876)。您可以通过在用于创建JAXBContext
的类列表中明确包含此类来解决此问题。
package forum10174513;
import java.io.File;
import java.util.*;
import javax.xml.bind.*;
import org.eclipse.persistence.jaxb.JAXBContextFactory;
public class Demo {
public static void main(String[] args) throws Exception {
Map<String, Object> properties = new HashMap<String, Object>(1);
properties.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, "forum10174513/bindings.xml");
JAXBContext jc = JAXBContext.newInstance(new Class[] {Root.class, BaseBeanEx.class}, properties);
File xml = new File("src/forum10174513/input.xml");
Unmarshaller unmarshaller = jc.createUnmarshaller();
Root root = (Root) unmarshaller.unmarshal(xml);
}
}
的 BaseBean 强> 的
package forum10174513;
public class BaseBean {
}
的 BaseBeanEx 强> 的
package forum10174513;
import javax.xml.bind.Unmarshaller;
public class BaseBeanEx extends BaseBean {
public void afterUnmarshal(Unmarshaller unmarshaller, Object parent) {
System.out.println("AFTER UNMARSHAL WAS CALLED");
}
}
的输出强> 的
以下是运行演示代码生成的输出。
AFTER UNMARSHAL WAS CALLED
AFTER UNMARSHAL WAS CALLED
了解更多信息
答案 1 :(得分:1)
更新:
更多地考虑它,jaxb可能永远不会发现回调,因为它不知道你的自定义子类。我假设JAXB在JAXBContext设置期间检查所有类。那时,JAXB只知道基础bean类,而不是自定义子类,因此永远不会找到回调方法。
2个想法。您可以使用“外部回调”机制(使用单独的事件处理程序来完成您的自定义类所需的操作)。或者,您可以尝试使用回调方法生成(或稍后添加)基本bean类。那么JAXB可能会识别并调用这些方法,然后您可以在自定义子类中覆盖这些方法。