2

我尝试用 MOXy准确地完成这个问题中描述的内容。但我有一些不同的行为,但结果是一样的。它不起作用。

我使用@XmlElements,并为每个具体类添加了一个带有名称和类型的@XmlElement。对于没有适配器的所有类,它都可以完美运行,但不适用于带有 PolyAdapter 的类。

在我的情况下,将调用适配器来编组对象。但它所写的只是一个像这样的条目到 xml 文件中。

<?xml version="1.0" encoding="UTF-8"?>
<fooContainer>
  <poly PolyName="PolyN$1">ABCBCD</poly>
  <cPlexPoly>moxy.CPlexPoly$CPlexPolyAdapter$CCPWrapper@480bdb19</cPlexPoly>
</fooContainer>

并且 unmarshaller 根本不会被调用。

我在这两种情况下测试了 MOXy 2.5.0 和当前的 RC3 2.5.1,结果是一样的。我还启用了调试输出,没有警告或错误。

或者也许还有另一种方法可以让这个工作

更新 1:提供模型

我创建了一个小例子来展示我对 MOXy 2.5.0 的问题

界面:

@XmlRootElement
public interface Foo {
  public List<String> getCoordinates();
}

没有 XmlAdapter 的类:

@XmlRootElement(name="poly")
public class Poly implements Foo {
  @XmlAttribute
  String PolyName = "PolyN$1";

  @XmlAnyElement
  private List<String> coordinates = new LinkedList<>();

  @Override
  public List<String> getCoordinates() {
    return coordinates;
  }

  public void addCoordinate( final String poly ) {
    coordinates.add( poly );
  }
}

带有 XmlAdapter 的类:

@XmlJavaTypeAdapter(CPlexPolyAdapter.class)
public class CPlexPoly implements Foo {
private Map<String, String> values = new HashMap<>();

@Override
public List<String> getCoordinates() {
  return new ArrayList<>(values.values());
}

public void addCoordinate( final String poly ) {
  values.put( Integer.toString( poly.hashCode()), poly );
}

public List<String> getKeys() {
  return new ArrayList<>(values.keySet());
}


public static class CPlexPolyAdapter extends XmlAdapter<CCPWrapper, CPlexPoly> {

  @XmlRootElement(name="wrapper")
  public static class CCPWrapper {
    @XmlAnyElement
    List<String> keys = new ArrayList<>();
    @XmlAnyElement
    List<String> values = new ArrayList<>();
  }

  @Override
  public CPlexPoly unmarshal(CCPWrapper v) throws Exception {
    // Won't be called. 
    return null;
  }

  @Override
  public CCPWrapper marshal(CPlexPoly v) throws Exception {
    if( v == null ) {
      return null;
    }

    CCPWrapper wrapper = new CCPWrapper();
    wrapper.values.addAll( v.getCoordinates() );
    wrapper.keys.addAll( v.getKeys() );

    return wrapper;
  } 
}
}

简单的容器类:

@XmlRootElement(name="fooContainer") 
public class FooContainer {

  @XmlElements({
    @XmlElement(name = "poly", type = Poly.class),
    @XmlElement(name = "cPlexPoly", type = CPlexPoly.class)
  })
  public List<Foo> container = new LinkedList<>();
}

主要的:

public class Main {
  public static void main(String[] args) {
    FooContainer fc = new FooContainer();

    Poly poly = new Poly();
    poly.addCoordinate("ABC");
    poly.addCoordinate("BCD");

    CPlexPoly cpoly = new CPlexPoly();
    cpoly.addCoordinate("XYZ");
    cpoly.addCoordinate("WXY");

    fc.container.add( poly );
    fc.container.add( cpoly );

    /* Marshall */
    JAXBContext context;
    try {
      context = JAXBContext.newInstance( FooContainer.class );
      Marshaller marshaller = context.createMarshaller();
      marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
      marshaller.marshal( fc, new FileWriter("fc.xml") );
    } 
    catch (JAXBException e) {
      e.printStackTrace();
    } catch (IOException e) {
      e.printStackTrace();
    }

    /* UnMarshall */
  }
}

我还更新了我上面描述的结果。谢谢。

更新 2:简化的包装器

即使这个简单的包装器只包含一个属性,也没有任何变化。

@XmlJavaTypeAdapter(CPlexPolyAdapter.class)
public class CPlexPoly implements Foo {

  private Map<String, String> values = new HashMap<>();

  @Override
  public List<String> getCoordinates() {
    return new ArrayList<>(values.values());
  }

  public void addCoordinate( final String poly ) {
    values.put( Integer.toString( poly.hashCode()), poly );
  }

  public List<String> getKeys() {
    return new ArrayList<>(values.keySet());
  }


  public static class CPlexPolyAdapter extends XmlAdapter<CCPWrapper, CPlexPoly> {

    @XmlRootElement(name="wrapper")
    public static class CCPWrapper {

      @XmlAttribute
      String test = "test";
    }

    @Override
    public CPlexPoly unmarshal(CCPWrapper v) throws Exception {
      // Won't be called. 
      return null;
    }

    @Override
    public CCPWrapper marshal(CPlexPoly v) throws Exception {
      if( v == null ) {
        return null;
      }

      CCPWrapper wrapper = new CCPWrapper();
      return wrapper;
    }
  }
}

因此,如果有人知道出了什么问题,那将非常有帮助。谢谢克里斯蒂安

4

1 回答 1

1

注意: 我是EclipseLink JAXB (MOXy)负责人,也是JAXB (JSR-222)专家组的成员。

您遇到的问题是由于CCPWrapper类映射不正确。您不允许将两个字段/属性映射为@XmlAnyElement.

    @XmlRootElement(name = "wrapper")
    public static class CCPWrapper {
        @XmlAnyElement
        List<String> keys = new ArrayList<>();
        @XmlAnyElement
        List<String> values = new ArrayList<>();
    }

当我修改您的代码以删除其中一个@XmlAnyElement注释时,一切似乎都正常工作。


更新#1

谢谢您的回答。但不,我仍然得到相同的输出。我什至用 XmlAnyElement 注释删除了这两个列表,只添加了一个简单的 XmlAttribute。输出不变。请参阅有问题的更新后的 Wrapper 类。

我只是注意到,当我尝试您的代码时,我已向该类添加了一个类型CPlexPoly字段FooContainer。除了我之前的建议之外,这还导致将XmlAdapter其应用于该container领域。

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

@XmlRootElement(name = "fooContainer")
public class FooContainer {

    @XmlElement
    private final CPlexPoly workaround = null;

    @XmlElements({ 
        @XmlElement(name = "poly", type = Poly.class),
        @XmlElement(name = "cPlexPoly", type = CPlexPoly.class) 
    })
    public List<Foo> container = new LinkedList<>();

}

您可以做一些类似的解决方法。我打开了以下错误,您可以使用它来跟踪我们在实际修复中的进度:


更新#2

更好的解决方法是将类包含CCPWrapper在用于创建的类中JAXBContext,如下所示:

    context = JAXBContext.newInstance( FooContainer.class, CCPWrapper.class );
于 2013-10-11T19:05:39.573 回答