XmlJavaTypeAdapter未被检测到。

9

希望对JAXB专家来说很简单:

我正在尝试编组一个不定义默认无参构造函数的不可变类。 我已经定义了一个XmlAdapter实现,但似乎没有被选中。 我已经准备了一个简单的自包含示例,但仍然无法正常工作。 请问我做错了什么?

不可变类

@XmlJavaTypeAdapter(FooAdapter.class)
@XmlRootElement
public class Foo {
  private final String name;
  private final int age;

  public Foo(String name, int age) {
    this.name = name;
    this.age = age;
  }

  public String getName() { return name; }
  public int getAge() { return age; }
}

适配器和值类型

public class FooAdapter extends XmlAdapter<AdaptedFoo, Foo> {
  public Foo unmarshal(AdaptedFoo af) throws Exception {
    return new Foo(af.getName(), af.getAge());
  }

  public AdaptedFoo marshal(Foo foo) throws Exception {
    return new AdaptedFoo(foo);
  }
}

class AdaptedFoo {
  private String name;
  private int age;

  public AdaptedFoo() {}

  public AdaptedFoo(Foo foo) {
    this.name = foo.getName();
    this.age = foo.getAge();
  }

  @XmlAttribute
  public String getName() { return name; }
  public void setName(String name) { this.name = name; }

  @XmlAttribute
  public int getAge() { return age; }
  public void setAge(int age) { this.age = age; }
}

Marshaller

public class Marshal {
  public static void main(String[] args) {
    Foo foo = new Foo("Adam", 34);

    try {
      JAXBContext jaxbContext = JAXBContext.newInstance(Foo.class);
      Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

      // output pretty printed
      jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

      jaxbMarshaller.marshal(foo, System.out);              
    } catch (JAXBException e) {
      e.printStackTrace();
    }   
  }
}

堆栈跟踪

com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException: 1 counts of IllegalAnnotationExceptions
Foo does not have a no-arg default constructor.
        this problem is related to the following location:
                at Foo

        at com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException$Builder.check(IllegalAnnotationsException.java:91)
        at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.getTypeInfoSet(JAXBContextImpl.java:451)
        at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.<init>(JAXBContextImpl.java:283)
        at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl.<init>(JAXBContextImpl.java:126)
        at com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl$JAXBContextBuilder.build(JAXBContextImpl.java:1142)
        at com.sun.xml.internal.bind.v2.ContextFactory.createContext(ContextFactory.java:130)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
        at java.lang.reflect.Method.invoke(Method.java:601)
        at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:248)
        at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:235)
        at javax.xml.bind.ContextFinder.find(ContextFinder.java:445)
        at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:637)
        at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:584)
        at Marshal2.main(Marshal2.java:11)

请注意,我正在使用JDK 1.7.0_05版本。
2个回答

8
以下内容应该有所帮助: 将Foo作为根对象 当在类型级别指定@XmlJavaTypeAdapter时,它仅适用于引用该类的字段/属性,而不是当该类的实例为XML树中的根对象时。这意味着您需要自己将Foo转换为AdaptedFoo,并在AdaptedFoo上创建JAXBContext,而不是Foo。 Marshal
package forum11966714;

import javax.xml.bind.*;

public class Marshal {
    public static void main(String[] args) {
      Foo foo = new Foo("Adam", 34);

      try {
        JAXBContext jaxbContext = JAXBContext.newInstance(AdaptedFoo.class);
        Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

        // output pretty printed
        jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

        jaxbMarshaller.marshal(new AdaptedFoo(foo), System.out);              
      } catch (JAXBException e) {
        e.printStackTrace();
      }   
    }
  }

AdaptedFoo

您需要在 AdaptedFoo 类上添加 @XmlRootElement 注释。您可以从 Foo 类中删除相同的注释。

package forum11966714;

import javax.xml.bind.annotation.*;

@XmlRootElement
class AdaptedFoo {
    private String name;
    private int age;

    public AdaptedFoo() {
    }

    public AdaptedFoo(Foo foo) {
        this.name = foo.getName();
        this.age = foo.getAge();
    }

    @XmlAttribute
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @XmlAttribute
    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

嵌套的对象作为Foo

Foo不是根对象时,一切都按照您映射的方式运作。我扩展了您的模型以演示它是如何工作的。

巴(Bar)

package forum11966714;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Bar {

    private Foo foo;

    public Foo getFoo() {
        return foo;
    }

    public void setFoo(Foo foo) {
        this.foo = foo;
    }

}

演示

需要注意的是,在启动JAXBContext时,JAXB参考实现不允许您指定Foo类。

package forum11966714;

import java.io.File;

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

public class Demo {
    public static void main(String[] args) {
        try {
            JAXBContext jaxbContext = JAXBContext.newInstance(Bar.class);

            Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller();
            File xml = new File("src/forum11966714/input.xml");
            Bar bar = (Bar) jaxbUnmarshaller.unmarshal(xml);

            Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
            jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
            jaxbMarshaller.marshal(bar, System.out);
        } catch (JAXBException e) {
            e.printStackTrace();
        }
    }
}

input.xml/Output

<?xml version="1.0" encoding="UTF-8"?>
<bar>
    <foo name="Jane Doe" age="35"/>
</bar>

1
谢谢Blaise!(我想这是你的博客,我最初看的)。看起来问题出在我在引导期间指定了Foo.class到JAXBContext。遗憾的是JAXB此时并没有意识到我已经为Foo指定了适配器...这使得引导有点棘手。 - Adamski
@bdoughan 您似乎是这个关于编组/解组的难题的专家,也许您可以在我的问题中给我一些指导。我有一个XML,它有一个名为<ROOT>的根元素和很多相同类型的元素,例如<ROOT><elem1 attr="value"</elem1><<elem2 attr="value"</elem2><elem3 attr="value"</elem3></ROOT>等。我正在尝试将其映射到一个Map<String,MyType> root对象上,以使这些元素成为动态元素。但是我迄今为止没有成功。我编写了一个@XmlJavaTypeAdapter,但它并没有像源问题中那样被选中。提前致谢。 - Frankie Drake

0

我知道实际上不是这种情况,但如果在字段中使用 @XmlJavaTypeAdapter 时出现此类错误,请检查是否指定了命名空间。你可能需要它。

在我的情况下,这种方法并没有起作用:

@XmlElement(name = "Expiration")
@XmlJavaTypeAdapter(DateAdapter.class)
private Date expiration;

直到指定了命名空间:

@XmlElement(name = "Expiration", namespace="http://site/your.namespace")
@XmlJavaTypeAdapter(DateAdapter.class)
private Date expiration;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接