How to add XML processing instructions during JAXB marshal

前端 未结 1 567
没有蜡笔的小新
没有蜡笔的小新 2021-02-09 05:22

I would like to add a processing instruction whenever a collection/array property is serialized to get something like


  
  <         


        
相关标签:
1条回答
  • 2021-02-09 05:45

    You could leverage an XMLStreamWriter and an XmlAdapter to do this:

    BobAdapter

    Things to note about the XmlAdapter:

    • It's stateful and references an XMLStreamWriter. We will later leverage JAXB's ability to set a stateful XmlAdapter on a Marshaller.
    • It converts from a List<String> to a List<String>, we're just using an XmlAdapter here to get an event.
    • The marshal method is where we call writeProcessingInstruction on the XMLStreamWriter:

     

    package forum6931520;
    
    import java.util.List;
    
    import javax.xml.bind.annotation.adapters.XmlAdapter;
    import javax.xml.stream.XMLStreamWriter;
    
    public class BobAdapter extends XmlAdapter<List<String>, List<String>> {
    
        private boolean first = true;
        private XMLStreamWriter xmlStreamWriter;
    
        public BobAdapter() {
        }
    
        public BobAdapter(XMLStreamWriter xmlStreamWriter) {
            this();
            this.xmlStreamWriter = xmlStreamWriter;
        }
    
        @Override
        public List<String> marshal(List<String> stringList) throws Exception {
            if(first) {
                xmlStreamWriter.writeProcessingInstruction("array", "bob");
                first = false;
            }
            return stringList;
        }
    
        @Override
        public List<String> unmarshal(List<String> stringList) throws Exception {
            return stringList;
        }
    
    }
    

    Alice

    The @XmlJavaTypeAdapter annotation is used to link the XmlAdapter with the field/property:

    package forum6931520;
    
    import java.util.List;
    
    import javax.xml.bind.annotation.XmlRootElement;
    import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
    
    @XmlRootElement
    public class Alice {
    
        private List<String> bob;
    
        @XmlJavaTypeAdapter(BobAdapter.class)
        public List<String> getBob() {
            return bob;
        }
    
        public void setBob(List<String> bob) {
            this.bob = bob;
        }
    
    }
    

    Demo

    package forum6931520;
    
    import java.io.File;
    
    import javax.xml.bind.JAXBContext;
    import javax.xml.bind.Marshaller;
    import javax.xml.bind.Unmarshaller;
    import javax.xml.stream.XMLOutputFactory;
    import javax.xml.stream.XMLStreamWriter;
    
    public class Demo {
    
        public static void main(String[] args) throws Exception {
            JAXBContext jc = JAXBContext.newInstance(Alice.class);
    
            File xml = new File("src/forum6931520/input.xml");
            Unmarshaller unmarshaller = jc.createUnmarshaller();
            Alice alice = (Alice) unmarshaller.unmarshal(xml);
    
            Marshaller marshaller = jc.createMarshaller();
            XMLOutputFactory xof = XMLOutputFactory.newFactory();
            XMLStreamWriter xmlStreamWriter = xof.createXMLStreamWriter(System.out);
            marshaller.setAdapter(new BobAdapter(xmlStreamWriter));
            marshaller.marshal(alice, xmlStreamWriter);
        }
    
    }
    

    input.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <alice>
      <?array bob?>
      <bob>edgar</bob>
      <bob>david</bob>
    </alice>
    

    Output

    <?xml version='1.0' encoding='UTF-8'?><alice><?array bob?><bob>edgar</bob><bob>david</bob></alice>
    

    Note

    This example needs to be run with EclipseLink JAXB (MOXy) as the JAXB RI will throw the following exception (I'm the MOXy lead):

    Exception in thread "main" com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException: 2 counts of IllegalAnnotationExceptions
    java.util.List is an interface, and JAXB can't handle interfaces.
        this problem is related to the following location:
            at java.util.List
            at public java.util.List forum6931520.Alice.getBob()
            at forum6931520.Alice
    java.util.List does not have a no-arg default constructor.
        this problem is related to the following location:
            at java.util.List
            at public java.util.List forum6931520.Alice.getBob()
            at forum6931520.Alice
    

    For More Information

    • http://blog.bdoughan.com/2010/07/xmladapter-jaxbs-secret-weapon.html
    • http://blog.bdoughan.com/2011/05/specifying-eclipselink-moxy-as-your.html

    UPDATE

    I have entered an enhancement request (https://bugs.eclipse.org/354286) to add native support for processing instructions. This would eventually allow you to specify the following on your property:

    @XmlProcessingInstruction(target="array", value="bob")
    public List<String> getBob() {
        return bob;
    }
    
    0 讨论(0)
提交回复
热议问题