JAXB / MOXy适应XMLElementWrapper

时间:2013-10-23 13:05:49

标签: java xml jaxb moxy

我试图使用JAXB / MOXy映射由一组具有独特特征的接口表示的域:所有多值元素都使用Iterables封装而不是数组或Collection秒。下面是一个展示我案例的玩具示例。

Actor.java(注意使用Iterable<Movie>

package test.moxy;

public interface Actor {

    public String getName();
    public void setName(String name);

    public Iterable<Movie> getMovies();
    public void setMovies(Iterable<Movie> movies);

}

Movie.java

package test.moxy;

public interface Movie {

    public String getTitle();
    public void setTitle(String title);

}

主要课程

package test.moxy;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

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

import org.eclipse.persistence.jaxb.JAXBContextProperties;
import org.xml.sax.InputSource;

public class TestBinding {

    public static void main(String[] args) throws FileNotFoundException, JAXBException {

        Class<?>[] types = new Class<?>[] { Actor.class, Movie.class };

        List<String> mappings = new ArrayList<String>();
        mappings.add("test/moxy/oxm.xml");

        Map<String, Object> properties = new HashMap<String, Object>();
        properties.put(JAXBContextProperties.OXM_METADATA_SOURCE, mappings);

        JAXBContext jc = JAXBContext.newInstance(types, properties);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        InputSource is = new InputSource(new FileInputStream("src/main/java/test/moxy/input.xml"));
        Actor actor = (Actor) unmarshaller.unmarshal(is);

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

    }

}

input.xml中

<?xml version="1.0" encoding="UTF-8"?>
<actor>
    <name>John Smith</name>
    <movies>
        <movie>
            <title>Smith's Trilogy - Part I</title>
        </movie>
        <movie>
            <title>Smith's Trilogy - Part II</title>
        </movie>
        <movie>
            <title>Smith's Trilogy - Part III</title>
        </movie>
    </movies>
</actor>

oxm.xml

<?xml version="1.0" encoding="UTF-8"?>
<xml-bindings xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
    package-name="test.moxy" xml-mapping-metadata-complete="true">
    <java-types>
        <java-type name="Actor">
        <xml-root-element name="actor" />
            <xml-type prop-order="name movies" factory-class="test.moxy.ProxyFactory"
                factory-method="initActor" />
            <java-attributes>
                <xml-element java-attribute="name" />
                <xml-element java-attribute="movies" name="movie" >
                    <xml-element-wrapper name="movies" />
                </xml-element>
            </java-attributes>
        </java-type>
        <java-type name="Movie">
            <xml-root-element name="movie" />
            <xml-type factory-class="test.moxy.ProxyFactory"
                factory-method="initMovie" />
            <java-attributes>
                <xml-element java-attribute="title" />
            </java-attributes>
        </java-type>
    </java-types>
</xml-bindings>

ProxyFactory是一个标准工厂,例如http://blog.bdoughan.com/2010/07/moxy-jaxb-map-interfaces-to-xml.html

中举例说明的工厂

按照目前的情况,我得到以下JAXBException:异常说明:XmlElementWrapper仅允许在集合或数组属性上,但[movies]不是集合或数组属性。

我尝试使用XMLAdapter电影在Iterable<T>List<T>之间转换,但它似乎适用于每个movie元素,而不是{{1包装器。我想知道是否以及如何为包装器指定某种movies这样的东西?

1 个答案:

答案 0 :(得分:0)

如异常所述,您无法使用Iterable映射@XmlElementWrapper类型的媒体资源。

Exception in thread "main" javax.xml.bind.JAXBException: 
Exception Description: XmlElementWrapper is only allowed on a collection or array property but [movies] is not a collection or array property.
 - with linked exception:
[Exception [EclipseLink-50015] (Eclipse Persistence Services - @VERSION@.@QUALIFIER@): org.eclipse.persistence.exceptions.JAXBException
Exception Description: XmlElementWrapper is only allowed on a collection or array property but [movies] is not a collection or array property.]
    at org.eclipse.persistence.jaxb.JAXBContext$TypeMappingInfoInput.createContextState(JAXBContext.java:1068)
    at org.eclipse.persistence.jaxb.JAXBContext.<init>(JAXBContext.java:182)

XmlAdapter(MoviesAdapter)

相反,我们将使用XmlAdapterIterable转换为我们可以映射的内容。在这种情况下,我们将组成一个名为Movies的类的实例,其List个实例为Movie

package test.moxy;

import java.util.*;
import javax.xml.bind.annotation.adapters.XmlAdapter;

public class MoviesAdapter extends XmlAdapter<MoviesAdapter.Movies, Iterable<Movie>> {

    public static class Movies {
        public List<Movie> movie = new ArrayList<Movie>();
    }

    @Override
    public Iterable<Movie> unmarshal(Movies v) throws Exception {
        return v.movie;
    }

    @Override
    public Movies marshal(Iterable<Movie> v) throws Exception {
        Movies movies = new Movies();
        for(Movie movie : v) {
            movies.movie.add(movie);
        }
        return movies;
    }

}

外部元数据(oxm.xml)

以下是引用oxm.xml的修改后的XmlAdapter。使用新映射时,Movies对象将导致movies元素出现,因此我们不再需要@XmlElementWrapper元数据。

<?xml version="1.0" encoding="UTF-8"?>
<xml-bindings xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
    package-name="test.moxy" xml-mapping-metadata-complete="true">
    <java-types>
        <java-type name="Actor">
        <xml-root-element name="actor" />
            <xml-type prop-order="name movies" factory-class="test.moxy.ProxyFactory"
                factory-method="initActor" />
            <java-attributes>
                <xml-element java-attribute="name" />
                <xml-element java-attribute="movies">
                    <xml-java-type-adapter value="test.moxy.MoviesAdapter"/>
                </xml-element>
            </java-attributes>
        </java-type>
        <java-type name="Movie">
            <xml-root-element name="movie" />
            <xml-type factory-class="test.moxy.ProxyFactory"
                factory-method="initMovie" />
            <java-attributes>
                <xml-element java-attribute="title" />
            </java-attributes>
        </java-type>
    </java-types>
</xml-bindings>