java - 基于对象属性的 JAXB 元素名称

标签 java jaxb quickbooks

我必须为以下 XML 创建对象模型:

XML 示例 1:

<InvoiceAdd>
  <TxnDate>2009-01-21</TxnDate>
  <RefNumber>1</RefNumber>
  <InvoiceLineAdd>
  </InvoiceLineAdd>
</InvoiceAdd>

XML 示例 2:

<SalesOrderAdd>
  <TxnDate>2009-01-21</TxnDate>
  <RefNumber>1</RefNumber>
  <SalesOrderLineAdd>
  </SalesOrderLineAdd>
</SalesOrderAdd>

XML 输出将基于单个字符串参数或枚举。 String txnType = "发票"; (或“销售订单”);

我会使用单类 TransactionAdd:

@XmlRootElement
public class TransactionAdd {  
  public String txnDate;
  public String refNumber;

  private String txnType;
  ...

  public List<LineAdd> lines;
}

而不是使用子类或其他任何东西。创建 TransactionAdd 实例的代码对于两种类型的交易都是相同的,只是类型不同。

此 XML 由一个名为 QuickBooks 的知名产品使用,并由 QuickBooks Web 服务使用 - 所以我无法更改 XML,但我希望能够轻松地根据属性 (txnType) 设置元素名称).

我会考虑类似确定目标元素名称的方法:

@XmlRootElement
public class TransactionAdd {  
  public String txnDate;
  public String refNumber;

  private String txnType;
  ...

  public List<LineAdd> lines;

  public String getElementName() {
     return txnType + "Add";
  }
}

将使用以下代码创建不同的交易:

t = new TransactionAdd();
t.txnDate = "2010-12-15";
t.refNumber = "123";
t.txnType = "Invoice";

目标是根据txnType用顶级元素名序列化t对象。例如:

<InvoiceAdd>
   <TxnDate>2009-01-21</TxnDate>
   <RefNumber>1</RefNumber>
</InvoiceAdd>

在 t.txnType = "SalesOrder"的情况下,结果应该是

<SalesOrderAdd>
   <TxnDate>2009-01-21</TxnDate>
   <RefNumber>1</RefNumber>
</SalesOrderAdd>

目前我只看到一个变通方法,使用子类 InvoiceAdd 和 SalesOrderAdd 并使用 @XmlElementRef 注释来获得基于类名的名称。但是它需要根据交易类型实例化不同的类,还需要有另外两个不同的类 InvoiceLineAdd 和 SalesOrderLineAdd,这看起来很丑陋。

请给我建议任何解决方案来处理这个问题。我会考虑一些简单的事情。

最佳答案

要解决根元素方面的问题,您需要利用@XmlRegistry 和@XmlElementDecl。这将为我们提供 TransactionAdd 类的多个可能的根元素:

import javax.xml.bind.JAXBElement;
import javax.xml.bind.annotation.XmlElementDecl;
import javax.xml.bind.annotation.XmlRegistry;
import javax.xml.namespace.QName;

@XmlRegistry
public class ObjectFactory {

    @XmlElementDecl(name="InvoiceAdd")
    JAXBElement<TransactionAdd> createInvoiceAdd(TransactionAdd invoiceAdd) {
        return new JAXBElement<TransactionAdd>(new QName("InvoiceAdd"), TransactionAdd.class, invoiceAdd);
    }

    @XmlElementDecl(name="SalesOrderAdd")
    JAXBElement<TransactionAdd> createSalesOrderAdd(TransactionAdd salesOrderAdd) {
        return new JAXBElement<TransactionAdd>(new QName("SalesOrderAdd"), TransactionAdd.class, salesOrderAdd);
    }

}

您的 TransactionAdd 类将如下所示。有趣的是,我们会将 txnType 属性设置为@XmlTransient。

import java.util.List;

import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlTransient;

public class TransactionAdd {

    private String txnDate;
    private String refNumber;
    private String txnType;
    private List<LineAdd> lines;

    @XmlElement(name="TxnDate")
    public String getTxnDate() {
        return txnDate;
    }

    public void setTxnDate(String txnDate) {
        this.txnDate = txnDate;
    }

    @XmlElement(name="RefNumber")
    public String getRefNumber() {
        return refNumber;
    }

    public void setRefNumber(String refNumber) {
        this.refNumber = refNumber;
    }

    @XmlTransient
    public String getTxnType() {
        return txnType;
    }

    public void setTxnType(String txnType) {
        this.txnType = txnType;
    }

    public List<LineAdd> getLines() {
        return lines;
    }

    public void setLines(List<LineAdd> lines) {
        this.lines = lines;
    }

}

然后我们需要在 JAXB 操作之外提供一些逻辑。对于解码,我们将使用根元素名称的本地部分来填充 txnType 属性。对于 marshal,我们将使用 txnType 属性的值来创建适当的 JAXBElement。

import java.io.File;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBElement;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;

public class Demo {

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

        File xml = new File("src/forum107/input1.xml");
        Unmarshaller unmarshaller = jc.createUnmarshaller();
        JAXBElement<TransactionAdd> je = (JAXBElement<TransactionAdd>) unmarshaller.unmarshal(xml);
        TransactionAdd ta = je.getValue();
        ta.setTxnType(je.getName().getLocalPart());

        JAXBElement<TransactionAdd> jeOut;
        if("InvoiceAdd".equals(ta.getTxnType())) {
            jeOut = new ObjectFactory().createInvoiceAdd(ta);
        } else {
            jeOut = new ObjectFactory().createSalesOrderAdd(ta);
        }
        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(jeOut, System.out);
    }

}

待办事项

接下来我将研究解决 lines 属性。

关于java - 基于对象属性的 JAXB 元素名称,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4441692/

相关文章:

java - 没有 WindowManager.LayoutParams.TYPE_PHONE 的粘性覆盖

java - JAXB2Marshaller 没有将 xml 元素解码为字符

java - 在 Ant 中从 JAXB 类文件生成 XML 模式

vb.net - QBSDK发票查询

java - 如何使用 Xpath 在 XML 树的节点后检索节点?

java - LinkedIn Wherehows - 如何启动后端?

java - 如何在能够暂停/恢复的同时以固定间隔从线程获取字符串值?

java - 实现 JAXBContext 解析器类 : What classes are stored in the list?

java - 用于连接 v3 中的 Quickbooks Desktop 的 API

quickbooks - 将基于 Web 的应用程序与 QuickBooks Online 集成