4

我编写了一个BaseBeanEx扩展 JAXB 注释 bean () 的 bean ( BaseBean)。它BaseBean位于数据结构中某处的列表中,无法更改。该软件会在BaseBeanEx需要时进行显式转换。我还写了一个ObjectFactoryto createBaseBeanEx而不是BaseBean. 这一切都很好,但现在我添加了一个永远不会被调用的afterUnmarshal方法。BaseBeanEx

这是一个错误还是根据规格?如果以后是这种情况,是否有一些优雅的解决方法?

我正在使用默认的 JAXB 引擎。

4

2 回答 2

5

注意: 我是EclipseLink JAXB (MOXy)负责人,也是JAXB 2 (JSR-222)专家组的成员。

afterUnmarshal没有被调用的原因是BaseBeanEx元数据是建立在BaseBean类上的。要让您的用例正常工作,您需要让您的 JAXB 实现知道您确实想要映射到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 提供程序,那么您可以利用其外部映射文档来应用元数据,而无需修改您的域模型。

绑定.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利用外部映射文档的 a。当前存在一个错误,其中仅通过外部映射文档引用的类不会注册事件方法 (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);
    }

}

基豆

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

了解更多信息

于 2012-04-16T14:31:55.043 回答
1

你拼写对了吗?方法名称是“afterUnmarshal”(一个“L”)。 眼镜

更新:

再想一想,jaxb 可能永远不会发现回调,因为它不知道您的自定义子类。我会假设 JAXB 在 JAXBContext 设置期间检查所有类。那时,JAXB 只知道基 bean 类,而不知道您的自定义子类,因此永远找不到回调方法。

2个想法。您可以使用“外部回调”机制(使用单独的事件处理程序来执行您的自定义类所需的操作)。或者,您可以尝试使用回调方法生成(或稍后添加)基 bean 类。那么 JAXB 可能会识别并调用这些方法,然后您可以在自定义子类中覆盖这些方法。

于 2012-04-16T13:16:19.713 回答