Only a handful of source code lines is required to make a JAXB Marshaller
object write a document tree as an XML file. First you obtain a Marshaller
from a JAXBContext
. Then, you might set a number of properties, such as the one that's used below, which requests nice formatting of the XML text. Other properties concern the inclusion of a schema location as an attribute in the top-level element, or the encoding in the XML prolog. The first argument must be an object that is either a root element, as defined by your schema, or a JAXBElement<?>
.
import java.io.*;
import javax.xml.bind.* void writeDocument( Object document, String pathname )
throws JAXBException, IOException {
Class<T> clazz = document.getValue().getClass();
JAXBContext context = JAXBContext.newInstance( clazz.getPackage().getName() );
Marshaller m = context.createMarshaller();
m.setProperty( Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE );
m.marshal( document, new FileOutputStream( pathname ) );
}
Sometimes marshalling needs to be done not only for one or two root documents but for objects of many different schema types. You could, of course, add xsd:element
definitions for all of them to the top level xsd:schema
element, but this is cumbersome. A generic solution is presented below. The method wraps an arbitrary element of some type T
into a JAXBElement<T>
.
<T> JAXBElement<T> wrap( String ns, String tag, T o ){
QName qtag = new QName( ns, tag );
Class<?> clazz = o.getClass();
@SuppressWarnings( "unchecked" )
JAXBElement<T> jbe = new JAXBElement( qtag, clazz, o );
return jbe;
}
To use it, you must create a context capable of handling all the classes that might crop up as instantiations for T
. (Creating a JAXBContext
for several packages or classes is explained in section The JAXB Context.) With a Marshaller m
obtained from this context, the application of wrap
is as simple as this:
SomeType st = ...;
JAXBElement<SomeType> jbx = wrap( "http://www.acme.com", "someTag", st );
m.marshal( jbx, System.out );