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 并且文件路径是正确的,只是为了隐私没有完全包含在上面的片段中。任何帮助将不胜感激。
请让您的 class 名称遵循 Pascal 表示法 - 这样更容易阅读和理解,尤其是当有很多代码时。也许,您应该更多地研究您的代码风格。
您的 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>
中的每个 元素都隐含地带有默认名称空间前缀。
序列化
我下载了您提供的 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 将从 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 并且文件路径是正确的,只是为了隐私没有完全包含在上面的片段中。任何帮助将不胜感激。
请让您的 class 名称遵循 Pascal 表示法 - 这样更容易阅读和理解,尤其是当有很多代码时。也许,您应该更多地研究您的代码风格。
您的
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>
中的每个 元素都隐含地带有默认名称空间前缀。
序列化
我下载了您提供的 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());
}
}
}
阅读理解