JAVA Bean和XML之间的相互转换 - XStream简单入门

背景介绍

我们在工作中经常 遇到文件解析为数据或者数据转化为xml文件的情况,之前我一直采用的方式都是手动的来解析或拼接XML文件,这个做法固然是可以的,但是代码部分会显得非常臃肿,所以就查找了相关的资料,发现了一个名为XStream的类库可以做这件事,下面我们来看一下.

XStream的简介

官网地址:http://x-stream.github.io/

从官网介绍我们能看到以下几个特性:

  • 易用
  • 不需要Mapping文件

    不需要额外提供Mapping文件即可完成映射

  • 高性能

    高速且低内存消耗

  • 整洁的XML文件

    生成的XML的相当简洁

  • 不需要修改对象

    可以序列化private修饰的属性,且不需要提供get/set方法

  • 完整的对象结构支持

    支持循环引用情况下的输出

  • 兼容其他的XML API

    可以对任何的树形结构进行序列化或反序列化

  • 可以自定义的转换策略

    可以通过策略的注册,允许自定义特定类型的转化为XML的样式

  • 安全的框架

    对未分组的类型进行精细化的控制,防止输入错误而导致的安全问题

  • 良好的异常反馈

    当XML文件格式错误的时候,能够提供诊断信息帮助处理

  • 多样的输出格式

    不止是XML他还支持JSON及其变种

注解简介

从XStream的官网我们了解到,XStream不需要Mapping文件就能完成JavaBean和XML之间的转变,但大部分时候我们需要对生成的XML文件进行定制,name我们需要了解XStream提供的注解

注解 作用 适用范围
@XStreamAlias 设置别名 类或属性
@XStreamImplicit 忽略属性 集合类型的属性
@XStreamAsAttribute 将JavaBean转换成节点属性 属性
@XStreamOmitField 忽略属性 属性
@XStreamConverter 注入转换器

应用实例

首先在我们的项目中加入依赖:

<dependency>
<groupId>com.thoughtworks.xstream</groupId>
<artifactId>xstream</artifactId>
<version>1.4.11.1</version>
</dependency>

假定环境为Computer和CPU两个类:

public class Computer implements Serializable {

    private static final long serialVersionUID = 567119860357020081L;

    private String name;
private String brand;
private List<CPU> processors; public Computer() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getBrand() {
return brand;
} public void setBrand(String brand) {
this.brand = brand;
} public List<CPU> getProcessors() {
return processors;
} public void setProcessors(List<CPU> processors) {
this.processors = processors;
}
}
public class CPU implements Serializable {

    private static final long serialVersionUID = -718438777162957779L;

    private String name;
private String generation; public CPU() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getGeneration() {
return generation;
} public void setGeneration(String generation) {
this.generation = generation;
}
}

写一个测试方法:

import com.hykj.facheck.entity.CPU;
import com.hykj.facheck.entity.Computer;
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.io.xml.DomDriver; import java.util.ArrayList;
import java.util.List; public class XStreamTest { public static void main(String[] args) { List<CPU> cpuList = new ArrayList<>();
CPU cpu1 = new CPU();
cpu1.setGeneration("8-i5-K");
cpu1.setName("第一个");
cpuList.add(cpu1); CPU cpu2 = new CPU();
cpu2.setGeneration("8-i5-U");
cpu2.setName("第二个");
cpuList.add(cpu2); Computer computer = new Computer();
computer.setBrand("huawei");
computer.setName("wej-laptop");
computer.setProcessors(cpuList); XStream xstream = new XStream(new DomDriver());
xstream.ignoreUnknownElements();
xstream.processAnnotations(Computer.class); //javabean to xml
String xml = xstream.toXML(computer);
System.out.println(xml); //xml to javabean
Computer computerFromXml = (Computer) xstream.fromXML(xml); System.out.println(computer.getName()); }
}

运行程序,控制台打结果如下:

Connected to the target VM, address: '127.0.0.1:53977', transport: 'socket'
<com.hykj.facheck.entity.Computer>
<name>wej-laptop</name>
<brand>huawei</brand>
<processors>
<com.hykj.facheck.entity.CPU>
<name>第一个</name>
<generation>8-i5-K</generation>
</com.hykj.facheck.entity.CPU>
<com.hykj.facheck.entity.CPU>
<name>第二个</name>
<generation>8-i5-U</generation>
</com.hykj.facheck.entity.CPU>
</processors>
</com.hykj.facheck.entity.Computer>
Security framework of XStream not initialized, XStream is probably vulnerable.
wej-laptop
Disconnected from the target VM, address: '127.0.0.1:53977', transport: 'socket' Process finished with exit code 0

也就是说这样就可以用了,很棒棒
但是这个XML和我们的需求不太一样,我们加上注解,代码变成了这样:

import com.thoughtworks.xstream.annotations.XStreamAlias;

import java.io.Serializable;
import java.util.List; @XStreamAlias("my")
public class Computer implements Serializable { private static final long serialVersionUID = 567119860357020081L; private String name;
private String brand;
@XStreamAlias("cpuList")
private List<CPU> processors; public Computer() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getBrand() {
return brand;
} public void setBrand(String brand) {
this.brand = brand;
} public List<CPU> getProcessors() {
return processors;
} public void setProcessors(List<CPU> processors) {
this.processors = processors;
}
}
import com.thoughtworks.xstream.annotations.XStreamAlias;

import java.io.Serializable;

@XStreamAlias("oneCpu")
public class CPU implements Serializable { private static final long serialVersionUID = -718438777162957779L; private String name;
private String generation; public CPU() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getGeneration() {
return generation;
} public void setGeneration(String generation) {
this.generation = generation;
}
}

测试方法不变,结果如下:

Connected to the target VM, address: '127.0.0.1:54181', transport: 'socket'
<my>
<name>wej-laptop</name>
<brand>huawei</brand>
<cpuList>
<oneCpu>
<name>第一个</name>
<generation>8-i5-K</generation>
</oneCpu>
<oneCpu>
<name>第二个</name>
<generation>8-i5-U</generation>
</oneCpu>
</cpuList>
</my>
Security framework of XStream not initialized, XStream is probably vulnerable.
wej-laptop
Disconnected from the target VM, address: '127.0.0.1:54181', transport: 'socket' Process finished with exit code 0


这个XML就很舒服啦.
我们载试一下循环引用的情况吧,代码如下:

import com.thoughtworks.xstream.annotations.XStreamAlias;

import java.io.Serializable;
import java.util.List; @XStreamAlias("my")
public class Computer implements Serializable { private static final long serialVersionUID = 567119860357020081L; private String name;
private String brand;
@XStreamAlias("cpuList")
private List<CPU> processors; public Computer() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getBrand() {
return brand;
} public void setBrand(String brand) {
this.brand = brand;
} public List<CPU> getProcessors() {
return processors;
} public void setProcessors(List<CPU> processors) {
this.processors = processors;
}
}
import com.thoughtworks.xstream.annotations.XStreamAlias;

import java.io.Serializable;

@XStreamAlias("oneCpu")
public class CPU implements Serializable { private static final long serialVersionUID = -718438777162957779L; private String name;
private String generation;
private Computer computer; public CPU() {
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getGeneration() {
return generation;
} public void setGeneration(String generation) {
this.generation = generation;
} public Computer getComputer() {
return computer;
} public void setComputer(Computer computer) {
this.computer = computer;
}
}
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.io.xml.DomDriver; import java.util.ArrayList;
import java.util.List; public class XStreamTest { public static void main(String[] args) { Computer computer = new Computer();
computer.setBrand("huawei");
computer.setName("wej-laptop"); List<CPU> cpuList = new ArrayList<>();
CPU cpu1 = new CPU();
cpu1.setGeneration("8-i5-K");
cpu1.setName("第一个");
cpu1.setComputer(computer);
cpuList.add(cpu1); CPU cpu2 = new CPU();
cpu2.setGeneration("8-i5-U");
cpu2.setName("第二个");
cpuList.add(cpu2);
cpu1.setComputer(computer);
computer.setProcessors(cpuList); XStream xstream = new XStream(new DomDriver());
xstream.ignoreUnknownElements();
xstream.processAnnotations(Computer.class); //javabean to xml
String xml = xstream.toXML(computer);
System.out.println(xml); //xml to javabean
Computer computerFromXml = (Computer) xstream.fromXML(xml); System.out.println(computer.getName()); }
}

运行一下,先查看是否构成循环引用:

确认有循环引用,继续运行代码,控制台打印如下:

Connected to the target VM, address: '127.0.0.1:54221', transport: 'socket'
<my>
<name>wej-laptop</name>
<brand>huawei</brand>
<cpuList>
<oneCpu>
<name>第一个</name>
<generation>8-i5-K</generation>
<computer reference="../../.."/>
</oneCpu>
<oneCpu>
<name>第二个</name>
<generation>8-i5-U</generation>
</oneCpu>
</cpuList>
</my>
Security framework of XStream not initialized, XStream is probably vulnerable.
wej-laptop
Disconnected from the target VM, address: '127.0.0.1:54221', transport: 'socket' Process finished with exit code 0

可以看到生成的XML中采用类似路径的reference属性来标识了位置,实际上我们一般是需要忽略这个属性的,采用@XStreamOmitField注解就好了.

JAVA Bean和XML之间的相互转换 - XStream简单入门的更多相关文章

  1. JAVA bean与XML互转的利器---XStream

    最近在项目中遇到了JAVA bean 和XML互转的需求, 本来准备循规蹈矩使用dom4j忽然想起来之前曾接触过的XStream, 一番研究豁然开朗,利器啊利器, 下来就XStream的一些用法与大家 ...

  2. JAXB实现java对象与xml之间转换

    JAXB简介: 1.JAXB能够使用Jackson对JAXB注解的支持实现(jackson-module-jaxb-annotations),既方便生成XML,也方便生成JSON,这样一来可以更好的标 ...

  3. 简单Java类与XML之间的转换

    需要的jar包:xmlpull_1_0_5.jar,xstream-1.4.1.jar) 1.工具类XstreamUtil package com.learn.util; import com.tho ...

  4. JSON-lib框架,JAVA对象与JSON、XML之间的相互转换

    Json-lib可以将Java对象转成json格式的字符串,也可以将Java对象转换成xml格式的文档,同样可以将json字符串转换成Java对象或是将xml字符串转换成Java对象. 一. 准备工作 ...

  5. 通过JAXB完成Java对象与XML之间的转换

    Java对象转换XML的过程叫marshal. XML转换到Java对象的过程叫unmarshal. 一.Java对象转化为XML 这里省略getter和setter方法 通过标注@XMLRootEl ...

  6. JAXB 实现java对象与xml之间互相转换

    首先熟悉一下JAXB实现对象与xml互转时常用的一些注解使用: 1.@XmlRootElement,用于类级别的注解,对应xml的跟元素.通过name属性定义这个根节点的名称. 2.@XmlAcces ...

  7. jaxb 专题一(JAXB 实现java对象与xml之间互相转换)

    首先熟悉一下JAXB实现对象与xml互转时常用的一些注解使用: 1.@XmlRootElement,用于类级别的注解,对应xml的跟元素.通过name属性定义这个根节点的名称. 2.@XmlAcces ...

  8. Java 里面各种类型之间的相互转换

    1.整形与字符型之间的数据类型转换: 一.int转换成char有两种方法: ①  是利用char的unicode编码 例:int num1 = 8; char ch1 = (char) (num1 + ...

  9. Java Bean与Map之间相互转化的实现

    目录树 概述 Apache BeanUtils将Bean转Map Apache BeanUtils将Map转Bean 理解BeanUtils将Bean转Map的实现之手写Bean转Map 概述 Apa ...

随机推荐

  1. node.js异步编程的几种模式

    Node.js异步编程的几种模式 以读取文件为例: 1.callback function const fs = require('fs'); //callback function fs.readF ...

  2. mysql null 相关bug

    select 1 = null 并不是预期的 False 而是 Null select 1 is null; 这样才会产生预期的 False select 1 not is null 这样写是 错误 ...

  3. python字符串 列表 元组 字典相关操作函数总结

    1.字符串操作函数 find 在字符串中查找子串,找到首次出现的位置,返回下标,找不到返回-1 rfind 从右边查找 join 连接字符串数组 replace 用指定内容替换指定内容,可以指定次数 ...

  4. Taro开发微信小程序之初始化地图到当前位置

    在componentDidMount中,初始化mapCtx. let _this = this this.mapCtx = Taro.createMapContext('container') //c ...

  5. python第一周语言基础

    控制语句 if语句,当条件成立时运行语句块.经常与else, elif(相当于else if) 配合使用. for语句,遍历列表.字符串.字典.集合等迭代器,依次处理迭代器中的每个元素. while语 ...

  6. linux学习笔记:linux常用的命令

    2018-11-19                                      常见命令快速查询一览表 命令 功能 ls 列出目录内容 cat 链接文件并打印到标准输出设备上(通常用来 ...

  7. python学习中遇到的错误及解决办法

    1. nodename nor servname provided 原因:Python程序中有段程序调用 socket.gethostbyname(socket.gethostname()) sock ...

  8. jqgrid修改表格内容为居中

    看了手册没有发现自带的方法,所以使用了自定义css <style> #tableDataSearch tr td{ text-align:center; } </style>

  9. spring注解式参数校验

    很痛苦遇到大量的参数进行校验,在业务中还要抛出异常或者返回异常时的校验信息,在代码中相当冗长,今天我们就来学习spring注解式参数校验. 其实就是:hibernate的validator. 开始啦. ...

  10. 在Linux 安装Python3.5.6详细文档!!!!

    在Linux 安装Python3.5.6详细文档!!!! 1.安装相关依赖库(工具包) yum install gcc patch libffi-devel python-devel  zlib-de ...