使用扩展类时未调用JAXB afterUnmarshall

时间:2012-04-16 12:56:41

标签: java xml jaxb unmarshalling moxy

我编写了一个扩展JAXB注释bean(BaseBeanEx)的bean(BaseBean)。 BaseBean位于数据结构中的某个列表中,无法更改。只要需要,软件会对BaseBeanEx进行显式转换。我还写了一个ObjectFactory来创建BaseBeanEx而不是BaseBean。这一切都运行正常,但现在我向afterUnmarshal添加了一个永远不会被调用的BaseBeanEx方法。

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

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

2 个答案:

答案 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)

你拼错了吗?方法名称是“afterUnmarshal”(一个'L')。 Specs

更新:

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

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