XStream - вызов добавления атрибута в дочерний элемент - PullRequest
0 голосов
/ 13 января 2020

У меня проблемы с добавлением атрибута в дочерний элемент для записи файла XML. Проблема, по крайней мере для меня, состоит в том, что POJO, с которым сопоставлен элемент, связан с родительским элементом. Я хочу, чтобы вывод XML выглядел так:

<TXLife version="2.42.00" xmlns="http://ACORD.org/Standards/Life/2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <TXLifeRequest>
    <TransRefGUID>906D67C1-CC4D-11CF-91FB-444554540000</TransRefGUID>
    <TransType tc="1203"></TransType>
    <TransExeDate>2020-01-06</TransExeDate>
    <TransExeTime>09:30:00</TransExeTime>
    ...multiple other parent and child elements..
  </TXLifeRequest>
</TXLife>

Это элемент TransType , в который мне нужно добавить атрибут (как показано выше).

POJOS:

@XStreamAlias("TXLife")
public class TXLife {

    @XStreamAsAttribute
    private String version;

    @XStreamAsAttribute
    final String xmlns = "http://ACORD.org/Standards/Life/2";

    @XStreamAsAttribute
    @XStreamAlias("xmlns:xsi")
    final String xlink="http://www.w3.org/2001/XMLSchema-instance";

    @XStreamImplicit
    private List<TXLifeRequest> transListings = new ArrayList<TXLifeRequest>();

    public TXLife() {
        super();
    }
...getters and setters
@XStreamAlias("TXLifeRequest")
public class TXLifeRequest {

    @XStreamAlias("TransRefGUID")
    private String transRefGUID;

    @XStreamAlias("TransType")
    private String transType;

    @XStreamAlias("TransExeDate")
    private String transExeDate;

    @XStreamAlias("TransExeTime")
    private String transExeTime;

    public TXLifeRequest(String transRefGUID, String transType, String transExeDate, String transExeTime) {
        this.transRefGUID = transRefGUID;
        this.transType = transType;
        this.transExeDate = transExeDate;
        this.transExeTime = transExeTime;
    }
...getters and setters

Конвертер:

public class TransTypeAttributeConverter implements Converter {

    @Override
    public void marshal(Object o, HierarchicalStreamWriter hierarchicalStreamWriter, MarshallingContext marshallingContext) {
        TXLifeRequest txLifeRequest = (TXLifeRequest) o;
        hierarchicalStreamWriter.addAttribute("tc", "1203");
        hierarchicalStreamWriter.setValue("");
    }

    @Override
    public Object unmarshal(HierarchicalStreamReader hierarchicalStreamReader, UnmarshallingContext unmarshallingContext) {
        return null;
    }

    @Override
    public boolean canConvert(Class type) {
        return type.equals(TXLifeRequest.class);
    }
}

Тестовый класс:

public class TestXMLOutput {

    public static void main(String[] args) throws UnsupportedEncodingException {

        TestXMLOutput testXMLOutput = new TestXMLOutput();

        TXLife txLife = new TXLife();
        txLife.setVersion("2.42.00");
        txLife.add(new TXLifeRequest("906D67C1-CC4D-11CF-91FB-444554540000", "",
                "2020-01-06", "09:30:00"));

        testXMLOutput.testConvertAndWriteXMLToFile(txLife);

    }

    public void testConvertAndWriteXMLToFile(TXLife txLife) throws UnsupportedEncodingException {
        XStream xstream = new XStream(new StaxDriver());
        xstream.processAnnotations(TXLife.class);
        xstream.alias("TXLife", TXLife.class);
        xstream.processAnnotations(TXLifeRequest.class);
        xstream.alias("TXLifeRequest", TXLifeRequest.class);

        BufferedOutputStream bos = null;
        try {
            bos = new BufferedOutputStream(new FileOutputStream("CITS_BoB/src/test/java/txlife.xml"));
        } catch (IOException e) {
            System.out.println("IOException Occurred: " + e.getMessage());
        }

        xstream.addImplicitCollection(TXLife.class, "transListings");
        //xstream.registerConverter(new TransTypeAttributeConverter());

        xstream.marshal(txLife, new PrettyPrintWriter(
                new OutputStreamWriter(bos, StandardCharsets.UTF_8)));


    }
}

Я обнаружил, что работает оба xstream.addImplicitCollection (...) и xstream.registerConverter (...) в тестовом классе создает следующий вывод XML:

<TXLife version="2.42.00" xmlns="http://ACORD.org/Standards/Life/2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <TXLifeRequest tc="1203"></TXLifeRequest>
</TXLife>

Я уверен, что я неправильно понял или вообще пропустил, как правильно использовать xstream для выполнения такого рода задач. В любых примерах, которые я обнаружил в сети, элемент, к которому добавлен атрибут, сопоставлен с классом, тогда как в моем случае элемент, к которому я хочу добавить атрибут, сам находится в классе, который сопоставлен с «родительским» элементом. Я прошел через веб-сайт xstream , но не нашел ни одного примера, имеющего непосредственное отношение к моей цели. JAXB, к сожалению, не вариант, так как я использую openjdk 12, который не поддерживает это. Я не знаю, есть ли писатель документов XML, встроенный в openjdk 12 (я все еще новичок в этой версии). Я думаю, что я не могу понять, как каким-то образом "связать" указанный дочерний элемент c в POJO TXLifeRequest с атрибутом, который я хочу добавить к нему. Прошу прощения за неправильное использование кода, например, строковые литералы, добавленные в ierarchicalStreamWriter.addAttribute ("t c", "1203"); в операторе TransTypeAttributeConverter I Я уверен, что находятся в неправильном контексте или месте. Если кто-то может пролить свет на это, это было бы очень ценно Спасибо!

Извините, я должен также добавить, что другие варианты, которые я рассмотрел, - это писать POJO для каждого элемента (который, как я понимаю, скорее всего экстремальный и неэффективный), либо сопоставлять дочерний элемент в POJO с внутренний класс такой, что он фактически становится его собственным родительским элементом.

1 Ответ

0 голосов
/ 15 января 2020

В итоге я использовал парсер DOM java в качестве решения, которое позволило мне установить атрибут непосредственно для элемента TransType . Если случайно кто-то знает , как решить эту проблему в XStream, мне все равно было бы интересно узнать. Приветствия.

...