JAXB 不检测元素

JAXB not detecting elements

我正在尝试利用 JAXB 将从 youtube rss 提要中获得的 xml 元素转换为对象。我似乎遵循了我所见过的大多数示例的结构,但仍然无法正常工作,因为提要中的列表似乎始终是空的。有谁知道如何解决这一问题? 这是我的 class供参考:

饲料 class:

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

@XmlRootElement(
    name = "feed",
    namespace = "http://www.w3.org/2005/Atom"
    )
    @XmlAccessorType (XmlAccessType.FIELD)
    public class feed {

    @XmlElement(name = "entry")
    private List<entry> entries;
    public List<entry> getEntry() {
        return this.entries;
    }
    public void setEntry(List<entry> entries) {
        this.entries = entries;
    }
}

条目class:

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

@XmlRootElement(name = "entry")
@XmlAccessorType(XmlAccessType.FIELD)
public class entry {
private String title, name, id, published;


public void settitle(String title){this.title = title;}
public String gettitle(){return title;}

public void setname(String name){this.name = name;};
public String getname() {
    return name;
}

public void setid(String id){this.id = id;}
public String getid() {
    return id;
}

public void setpublished(String published){this.published = published;}
public String getpublished() {return published;}


public void PrintVideoInfo(){
    System.out.println(gettitle());
    System.out.println(getname());
    System.out.println(getid());
    System.out.println(getpublished());
    System.out.println("-----------");
}
}

解组 class:

import java.io.File;
import java.util.List;

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

public class XMLtoObject {
public static void main(String[] args) {

    try {

        File file = new File("videos.xml");
        JAXBContext jaxbContext = JAXBContext.newInstance(feed.class);

        Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller();
        feed que= (feed) jaxbUnmarshaller.unmarshal(file);

        for(entry ent:que.getEntry())
            ent.PrintVideoInfo();

    } catch (JAXBException e) {
        e.printStackTrace();
    }

}
}

https://www.youtube.com/feeds/videos.xml?channel_id=UCBcRF18a7Qf58cCRy5xuWwQ 这是正在使用的 xml。它被保存为 videos.xml 并且文件路径是正确的,只是为了隐私没有完全包含在上面的片段中。任何帮助将不胜感激。

  1. 请让您的 class 名称遵循 Pascal 表示法 - 这样更容易阅读和理解,尤其是当有很多代码时。也许,您应该更多地研究您的代码风格。

  2. 您的 feed class 注释正确,但为了让 JAXB 将 XML 解组为条目 entry class 应该也被适当地注释。另请注意,您的字段名称与提供的 xml 文件中的标签名称不匹配(<name> 无法在 <feed> 中直接访问)。 因此,添加注释和 classes 以匹配 xml 文件结构。

更新 1

我意识到我不太了解xml文档,所以我做了一点调查。

如果您完全熟悉 xml 命名空间,那么您可以跳过这部分到序列化部分。否则,请继续阅读。

Namespace 是一种将 xml 节点划分为非相交集的机制。想象一下,在您的 xml 中有一个 <address> 标签,该标签被定义了很多次。例如,它可以指网址或街道地址,因此根据上下文具有完全不同的含义。为避免混淆,您可以添加这样的名称空间前缀:<web:address> <street:address> 以将它们分开并将引用相同名称空间的其他元素分组。您之前在某个根标记中定义了它们:

<root xmlns:web="Web"> 
    <web:address> ... </web:address>
</root>

此外,还有一个特殊的命名空间 - 默认命名空间 - 其定义如下:<feed xmlns="Name">。拥有默认名称空间允许您在每次定义 xml 元素时省略编写名称空间前缀。

让我们用您的 xml 示例来解决这个问题:它为 <feed> 元素声明了三个名称空间(yt、media 和 default 名称空间)

<feed xmlns:yt="http://www.youtube.com/xml/schemas/2015" 
      xmlns:media="http://search.yahoo.com/mrss/" 
      xmlns="http://www.w3.org/2005/Atom">

这意味着 <feed> 中的每个 元素都隐含地带有默认名称空间前缀。

Intro to XML namespaces

Atom namespace explained

序列化

我下载了您提供的 xml 文件并做了一些测试。事实证明,JAXB 只是 "didn't see" 实体标签,因为它们隐藏在默认的命名空间后面,我们从来没有对 JAXB 说过有一个命名空间,除了 <feed> 元素。

因此,解决方案是使用命名空间注释要反序列化的元素,以便 JAXB 能够理解。

更新 2

看起来好像上面提供的解决方案太混乱了:必须用名称空间注释每个其他元素确实违反了 DRY 原则。幸运的是,有一种解决方案可以在一行中添加一个默认命名空间。

创建一个名为 package-info.java 的文件并在其中添加以下内容并将 package 替换为您的:

@XmlSchema(
        namespace = "http://www.w3.org/2005/Atom",
        elementFormDefault = XmlNsForm.QUALIFIED
)
package package;

import javax.xml.bind.annotation.XmlNsForm;
import javax.xml.bind.annotation.XmlSchema;

它所做的只是为我们要解析的文档定义一个 xml 模式。您现在可以删除所有 namespace = "..." 行并美化代码 如果您不熟悉 xml 架构,请查看它,因为它是控制 xml 文档结构的好方法。

更新2后的代码:

供稿 class

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

@XmlRootElement(name = "feed")
@XmlAccessorType(XmlAccessType.FIELD)
  public class Feed {

    @XmlElement(name = "entry")
    private List<Entry> entries;

    public List<Entry> getEntries() {
        return this.entries;
    }

}

条目class

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

@XmlRootElement(name = "entry")
@XmlAccessorType(XmlAccessType.FIELD)
public class Entry {

    @XmlElement(name = "title")
    private String title;

    @XmlElement(name = "id")
    private String id;

    @XmlElement(name = "published")
    private Date datePublished;

    @XmlElement(name = "author")
    private Author author;

    public String toString(){
        return String.format("Id: %s, Title: %s, Author: %s, Published: %s",
               id,
               title,
               author.toString(),
               datePublished.toString());
        }

}

作者class

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement(name = "author")
@XmlAccessorType(XmlAccessType.FIELD)
public class Author {

    @XmlElement(name = "name")
    private String name;

    @XmlElement(name = "url")
    private String url;

    public String getName() {
        return name;
    }

    public String getUrl() {
        return url;
    }

    @Override
    public String toString() {
        return getName();
    }

}

主要

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import java.io.File;

public class Main {

    public static void main(String[] args) throws JAXBException {
        Feed feed = (Feed) JAXBContext
                .newInstance(Feed.class)
                .createUnmarshaller()
                .unmarshal(new File("youtube_feed.xml"));

        for (Entry entry : feed.getEntries()) {
            System.out.println(entry.toString());
        }

    }
}

阅读理解

JAXB and XML namespaces

Oracle JAXB annotations reference