JaxB读取类层次结构

时间:2013-04-01 09:56:30

标签: java xml-parsing jaxb

只是提出Parsing class hierarchy using JaxB问题。 想要使用JaxB

阅读以下xml文件
<IMPORT>
    <TABLE NAME="USER">
        <ROW>
            <USER_ID>1</USER_ID>
            <ROW_VERSION>1</ROW_VERSION>
            <USER_NAME>Navnath</USER_NAME>
            <LOGIN>Navnath</LOGIN>
            <LOGIN_PASSWORD>Navnath</LOGIN_PASSWORD>
        </ROW>
        <ROW>
            <USER_ID>2</USER_ID>
            <ROW_VERSION>1</ROW_VERSION>
            <USER_NAME>Kumbhar</USER_NAME>
            <LOGIN>Kumbhar</LOGIN>
            <LOGIN_PASSWORD>Kumbhar</LOGIN_PASSWORD>
        </ROW>
    </TABLE>

    <TABLE NAME="WORK">
        <ROW>
            <WORK_ID>1</WORK_ID>
            <WORK_NAME>Work1</WORK_NAME>
            <ROW_VERSION TYPE="N">1</ROW_VERSION>
        </ROW>
        <ROW>
            <WORK_ID>2</WORK_ID>
            <WORK_NAME>Work2</WORK_NAME>
            <ROW_VERSION TYPE="N">1</ROW_VERSION>
        </ROW>
    </TABLE>
    <TABLE> ... </TABLE>
    <TABLE> ... </TABLE>
    <TABLE> ... </TABLE>
</IMPORT>

你可以在上面的xml文件中看到每个表中的列名都不同。我想在数据库中插入这些数据。我尝试为此创建类层次结构,但我不知道如何做到这一点。我的ROW类将包含每个表的不同xml元素,这是我无法配置的区域。请建议。

1 个答案:

答案 0 :(得分:1)

注意:我是EclipseLink JAXB (MOXy)主管,是JAXB (JSR-222)专家组的成员。

您可以针对此用例使用MOXy的@XmlDescriminatorNode / @XmlDescriminatorValue扩展名(请参阅:http://blog.bdoughan.com/2010/11/jaxb-and-inheritance-moxy-extension.html)。

导入

import java.util.List;
import javax.xml.bind.annotation.*;

@XmlRootElement(name="IMPORT")
@XmlAccessorType(XmlAccessType.FIELD)
public class Import {

    @XmlElement(name="TABLE")
    private List<Table> tables;

}

表格

@XmlDescriminatorNode注释用于指定将用于指示将实例化哪个子类的XML属性。 JAXB实现不能通过反射引入类的子类,我们将使用@XmlSeeAlso注释来引用它们。

import javax.xml.bind.annotation.XmlSeeAlso;
import org.eclipse.persistence.oxm.annotations.XmlDiscriminatorNode;

@XmlDiscriminatorNode("@NAME")
@XmlSeeAlso({UserTable.class, WorkTable.class})
public abstract class Table {

}

<强>用户表

@XmlDescriminatorValue注释用于指定与特定子类对应的NAME属性的值。

import java.util.List;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlDiscriminatorValue;

@XmlDiscriminatorValue("USER")
@XmlAccessorType(XmlAccessType.FIELD)
public class UserTable extends Table {

    @XmlElement(name="ROW")
    private List<UserRow> rows;

}

<强> UserRow

import javax.xml.bind.annotation.XmlElement;

public class UserRow {

    @XmlElement(name="USER_ID")
    private int userID;

    @XmlElement(name="USER_NAME")
    private String userName;

}

<强>工作台

import org.eclipse.persistence.oxm.annotations.XmlDiscriminatorValue;

@XmlDiscriminatorValue("WORK")
public class WorkTable extends Table {

}

jaxb.properties

要将MOXy指定为JAXB提供程序,您需要在与域模型相同的程序包中包含名为jaxb.properties的文件,并带有以下条目(请参阅:http://blog.bdoughan.com/2011/05/specifying-eclipselink-moxy-as-your.html)。

javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory

<强>演示

import java.io.File;
import javax.xml.bind.*;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Import.class);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum15741264/input.xml");
        Import result = (Import) unmarshaller.unmarshal(xml);

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(result, System.out);
    }

}

<强> input.xml中/输出

以下是运行演示代码的输入和输出。

<?xml version="1.0" encoding="UTF-8"?>
<IMPORT>
   <TABLE NAME="USER">
      <ROW>
         <USER_ID>1</USER_ID>
         <USER_NAME>Navnath</USER_NAME>
      </ROW>
      <ROW>
         <USER_ID>2</USER_ID>
         <USER_NAME>Kumbhar</USER_NAME>
      </ROW>
   </TABLE>
   <TABLE NAME="WORK"/>
</IMPORT>

替代解决方案

或者仅使用标准JAXB API,您可以使用XmlAdapter

尝试以下方法