JAXB Unmarshal子类基于子元素值

时间:2015-10-16 14:38:23

标签: java jaxb moxy

我需要根据元素类型的值映射关注xml。 BaseEntity超类具有Machine和Robot类的公共元素。每个子类都有不同的元素...... XML结构是固定的,我无法改变它。 每个入口元素应该映射到相应的类,entry / type = Machine应该映射到子类Machine等等......

在JAXB中可以吗?我该如何实现呢? 有什么建议吗?

<root>
    <entries>
        <entry>
            <name>RTM</name>
            <description>RealTime Machine</description>
            <code>RTM1</code>
            <type>Machine</type>
        </entry>
        <entry>
            <name>RTM</name>
            <description>RealTime Machine</description>
            <type>Robot</type>
            <serial>RS123<serial>
        </entry>
    </entries>
</root>

public abstract class BaseEntity {
    private String name;
    private String description;
}
public class Machine extends BaseEntity{
    private String code;
}
public class Robot extends BaseEntity{
    private String serial;
}

1 个答案:

答案 0 :(得分:1)

不幸的是你的XML布局非常糟糕。当JAXB解析器进入入口节点时,它无法告知节点它是否是机器或机器人。所以它必须猜测哪种类型用于解组。通常,您将使用类型作为节点名称(1)或提供类型参数(2)的属性。

1:

<entries>
    <machine>...</machine>
    <robot>...</robot>
</entries>

2:

<entries>
    <entry type="machine">...</entry>
    <entry type="robot">...</entry>
</entries>

所以我对你的问题的解决方案就像你的XML一样frankenstein。

有一个改编的类,它扩展了基类并具有可用子类的所有属性:

@XmlRootElement(name = "ENTRY")
@XmlAccessorType(XmlAccessType.FIELD)
public class AdaptedBaseEntry extends BaseEntry
{


    @XmlElement(name = "CODE", required = false)
    public String code;

    @XmlElement(name = "SERIAL", required = false)
    public String serial;

    public AdaptedBaseEntry()
    {
        super();
    }

    public AdaptedBaseEntry(BaseEntry entry)
    {
        super(entry.type, entry.description, entry.name);
        if (entry instanceof Machine)
        {
            this.code = ((Machine) entry).code;
        } else if (entry instanceof Robot)
        {
            this.serial = ((Robot) entry).serial;
        }
    }

}

使用XmlAdapter将适配器类绑定到XML,反之亦然:

public class BaseEntryAdapter extends XmlAdapter<AdaptedBaseEntry, BaseEntry>
{

    @Override
    public BaseEntry unmarshal(AdaptedBaseEntry v) throws Exception
    {
        switch (v.type)
        {
        case Machine:
            return new Machine(v.name, v.description, v.code);
        case Robot:
            return new Robot(v.name, v.description, v.serial);
        default:
            return null;
        }
    }

    @Override
    public AdaptedBaseEntry marshal(BaseEntry v) throws Exception
    {
        return new AdaptedBaseEntry(v);
    }

}