本文讲解,如何使用Digester将XML文档和Java对象映射,主要以撸代码为主。
引入 Maven pom依赖:
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-digester3</artifactId>
<version>3.2</version>
<classifier>with-deps</classifier>
</dependency>
现有XML文件books.xml,其文件内容如下:<br >
<?xml version="1.0" encoding="utf-8" ?>
<books>
<book name="水浒传">
<author>
<name>施耐庵</name>
<name>罗贯中</name>
</author>
<creationDate>元末明初</creationDate>
<byname>
<name>水浒</name>
<name>忠义水浒传</name>
</byname>
<literaryStyle>小说</literaryStyle>
</book>
<book name="西游记">
<author>
<name>吴承恩</name>
</author>
<creationDate>明朝</creationDate>
<byname>
<name>西游释厄传</name>
</byname>
<literaryStyle>
浪漫主义章回体长篇神魔小说
</literaryStyle>
</book>
<book name="红楼梦">
<author>
<name>曹雪芹</name>
<name>程伟元</name>
<name>高鹗</name>
</author>
<creationDate>18世纪中叶</creationDate>
<byname>
<name>石头记</name>
<name>情僧录</name>
<name>风月宝鉴</name>
<name>金陵十二钗</name>
<name>金玉缘</name>
</byname>
<literaryStyle>
章回体长篇小说
</literaryStyle>
</book>
</books>
本文提供两种将XML映射成Java对象的方式,分别是使用注解和是不使用注解。
1.映射方式一不使用注解
创建对象用来装载XML文档中的内容:
1、Name.java,用来存放books/book/byname/name标签中的内容。
public class Name {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
2、Byname.java,用来存放books/book/byname/name标签集合。
public class Byname {
private List<Name> name;
public List<Name> getName() {
return name;
}
public void setName(List<Name> name) {
this.name = name;
}
/**
*/
public void addByname(Name name){
if(name==null){
return;
}
if(this.name==null){
this.name = new ArrayList<>();
}
this.name.add(name);
}
}
3、AuthorName.java,用来存放books/book/author/name标签中的内容。
public class AuthorName {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
4、Author.java,用来存放books/book/author/name标签集合
public class Author {
private List<AuthorName> names;
public List<AuthorName> getNames() {
return names;
}
public void setNames(List<AuthorName> names) {
this.names = names;
}
public void addAuthorName(AuthorName name) {
if (name == null) {
return;
}
if (this.names == null) {
this.names = new ArrayList<>();
}
this.names.add(name);
}
}
5、Book.java,用来存放books/book标签中的内容
public class Book {
private Author author;
private Byname byname;
private String creationDate;
private String literaryStyle;
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Byname getByname() {
return byname;
}
public void setByname(Byname byname) {
this.byname = byname;
}
public Author getAuthor() {
return author;
}
public void setAuthor(Author author) {
this.author = author;
}
public String getCreationDate() {
return creationDate;
}
public void setCreationDate(String creationDate) {
this.creationDate = creationDate;
}
public String getLiteraryStyle() {
return literaryStyle;
}
public void setLiteraryStyle(String literaryStyle) {
this.literaryStyle = literaryStyle;
}
public void addByname(Byname byname){
this.byname = byname;
}
public void addAuthor(Author author){
this.author = author;
}
}
6、Books.java,用来存放books/book标签集。
public class Books {
private List<Book> bookList;
public List<Book> getBookList() {
return bookList;
}
public void setBookList(List<Book> bookList) {
this.bookList = bookList;
}
public void addBooks(Book book) {
if (book == null) {
return;
}
if (this.bookList == null) {
this.bookList = new ArrayList<>();
}
this.bookList.add(book);
}
}
7、Test1.java,生成映射XML文档到Java对象和测试
public class Test1 {
public static Books parseXml(File xmlFile) throws IOException, SAXException {
Digester digester = new Digester();
digester.setValidating(false);
//跟标签/books和Books对象映射
digester.addObjectCreate("books", Books.class);
//标签/books/book和Book对象映射
digester.addObjectCreate("books/book", Book.class);
//将/books/book标签的所有属性映射到Book对象的属性上,在这里映射的是/books/book标签的name属性。
digester.addSetProperties("books/book");
//或者使用下面这种方式代替,digester.addSetProperties("books/book");这种方式要求标签属性名和对象中的字段要保持命名一致才可以映射上。
// digester.addSetProperties("books/book","name","name");
//标签books/book/creationDate,和Book对象的creationDate属性映射
digester.addBeanPropertySetter("books/book/creationDate", "creationDate");
//标签books/book/literaryStyle,和Book对象的literaryStyle属性映射
digester.addBeanPropertySetter("books/book/literaryStyle", "literaryStyle");
//标签books/book/author和Author对象映射
digester.addObjectCreate("books/book/author", Author.class);
//标签books/book/author/name和AuthorName对象映射
digester.addObjectCreate("books/book/author/name", AuthorName.class);
//标签books/book/author/name,和AuthorName对象的name属性映射
digester.addBeanPropertySetter("books/book/author/name", "name");
//标签books/book/byname和Byname对象映射
digester.addObjectCreate("books/book/byname", Byname.class);
//标签books/book/byname/name和Name对象映射
digester.addObjectCreate("books/book/byname/name", Name.class);
//标签books/book/byname/name,和Byname对象的name属性映射
digester.addBeanPropertySetter("books/book/byname/name", "name");
//把Book标签对象添加到Book对象中,需要保证Books对象中有addBooks该方法,用于添加装载XML标签内容后的对象信息
digester.addSetNext("books/book", "addBooks");
//把Author标签对象和Byname标签对象添加到Book对象中,需要保证Book对象中有addAuthor和addByname方法,用于添加装载XML标签内容后的对象信息
digester.addSetNext("books/book/author", "addAuthor");
digester.addSetNext("books/book/byname", "addByname");
//把Name标签对象添加到Byname对象中,需要保证Byname对象中有addByname方法(命名任意,只需要对应上即可),用于添加装载XML标签内容后的对象信息
digester.addSetNext("books/book/byname/name", "addByname");
//把AuthorName标签对象添加到Author对象中,需要保证Author对象中有addAuthorName方法(命名任意,只需要对应上即可),用于添加装载XML标签内容后的对象信息
digester.addSetNext("books/book/author/name", "addAuthorName");
Object obj = digester.parse(xmlFile);
if (obj instanceof Books) {
return (Books) obj;
}
return null;
}
public static void main(String[] args) throws IOException, SAXException {
String baseDir = System.getProperty("user.dir");
File xmlFile = new File(baseDir + "/src/main/resources/books.xml");
Books books = parseXml(xmlFile);
}
}
2.映射方式二使用注解
Digester3根据注解来映射Java对象和XML文档,在方式一的基础上的对象添加注解即可,具体如下:
1、Name.java,用来存放books/book/byname/name标签中的内容。
//books/book/byname/name标签和Name对象映射
@ObjectCreate(pattern = "books/book/byname/name")
public class Name {
//books/book/byname/name标签内容和AuthorName的name属性映射
@BeanPropertySetter(pattern = "books/book/byname/name")
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
2、Byname.java,用来存放books/book/byname/name标签集合。
//books/book/byname标签和Byname对象映射
@ObjectCreate(pattern = "books/book/byname")
public class Byname {
private List<Name> name;
public List<Name> getName() {
return name;
}
public void setName(List<Name> name) {
this.name = name;
}
//将books/book/byname/name标签的内容对象添加到Byname对象中
@SetNext
public void addByname(Name name){
if(name==null){
return;
}
if(this.name==null){
this.name = new ArrayList<>();
}
this.name.add(name);
}
}
3、AuthorName.java,用来存放books/book/author/name标签中的内容。
//books/book/author/anme标签和AuthorName对象映射
@ObjectCreate(pattern = "books/book/author/name")
public class AuthorName {
@BeanPropertySetter(pattern = "books/book/author/name")
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
4、Author.java,用来存放books/book/author/name标签中的内容。
//books/book/author标签和Author对象映射
@ObjectCreate(pattern = "books/book/author")
public class Author {
private List<AuthorName> names;
public List<AuthorName> getNames() {
return names;
}
public void setNames(List<AuthorName> names) {
this.names = names;
}
//将books/book/author/name标签的内容对象添加到Author对象中
@SetNext
public void addAuthorName(AuthorName name) {
if (name == null) {
return;
}
if (this.names == null) {
this.names = new ArrayList<>();
}
this.names.add(name);
}
}
5、Book.java,用来存放books/book标签中的内容
//books/book标签和Book对象映射
@ObjectCreate(pattern = "books/book")
public class Book {
private Author author;
private Byname byname;
//books/book/creationDate标签内容和Book的creationDate属性映射
@BeanPropertySetter(pattern = "books/book/creationDate")
private String creationDate;
//books/book/literaryStyle标签内容和Book的literaryStyle属性映射
@BeanPropertySetter(pattern = "books/book/literaryStyle")
private String literaryStyle;
//将/books/book标签的所有属性映射到Book对象的属性上,在这里映射的是/books/book标签的name属性。
@SetProperty(pattern = "books/book")
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Byname getByname() {
return byname;
}
public void setByname(Byname byname) {
this.byname = byname;
}
public Author getAuthor() {
return author;
}
public void setAuthor(Author author) {
this.author = author;
}
public String getCreationDate() {
return creationDate;
}
public void setCreationDate(String creationDate) {
this.creationDate = creationDate;
}
public String getLiteraryStyle() {
return literaryStyle;
}
public void setLiteraryStyle(String literaryStyle) {
this.literaryStyle = literaryStyle;
}
//将books/book/byname标签的内容对象添加到Book对象中
@SetNext
public void addByname(Byname byname){
this.byname = byname;
}
//将books/book/author标签的内容对象添加到Book对象中
@SetNext
public void addAuthor(Author author){
this.author = author;
}
}
6、Books.java,用来存放books/book标签集。
//books标签和Books对象映射
@ObjectCreate(pattern = "books")
public class Books {
private List<Book> bookList;
public List<Book> getBookList() {
return bookList;
}
public void setBookList(List<Book> bookList) {
this.bookList = bookList;
}
//将books/book标签的内容对象添加到Books对象中
@SetNext
public void addBooks(Book book) {
if (book == null) {
return;
}
if (this.bookList == null) {
this.bookList = new ArrayList<>();
}
this.bookList.add(book);
}
}
7、Test2.java测试Digester3注解映射XML和Java对象。
public class Test2 {
public static Books readBooks(File xmlPath, Class<?> XmlClazz) throws IOException, SAXException {
Digester digester = getLoader(XmlClazz).newDigester();
return digester.parse(xmlPath);
}
public static DigesterLoader getLoader(final Class<?> XmlClazz) {
return newLoader(new FromAnnotationsRuleModule() {
@Override
protected void configureRules() {
bindRulesFrom(XmlClazz);
}
});
}
public static void main(String[] args) throws IOException, SAXException {
String baseDir = System.getProperty("user.dir");
File xmlFile = new File(baseDir + "/src/main/resources/books.xml");
Books books = readBooks(xmlFile, Books.class);
}
}