序列化工具 XStream

XStream 是一个简单的基于 Java 库,Java 对象序列化到 XML,反之亦然(即:可以轻易的将 Java 对象和 xml 文档相互转换)。

特点

  • 使用方便 - XStream 的 API 提供了一个高层次外观,以简化常用的用例。
  • 无需创建映射 - XStream 的 API 提供了默认的映射大部分对象序列化。
  • 性能 - XStream 快速和低内存占用,适合于大对象图或系统。
  • 干净的XML - XStream 创建一个干净和紧凑 XML 结果,这很容易阅读。
  • 不需要修改对象 - XStream 可序列化的内部字段,如私有和最终字段,支持非公有制和内部类。默认构造函数不是强制性的要求。
  • 完整对象图支持 - XStream 允许保持在对象模型中遇到的重复引用,并支持循环引用。
  • 可自定义的转换策略 - 定制策略可以允许特定类型的定制被表示为XML的注册。
  • 安全框架 - XStream 提供了一个公平控制有关解组的类型,以防止操纵输入安全问题。
  • 错误消息 - 出现异常是由于格式不正确的XML时,XStream 抛出一个统一的例外,提供了详细的诊断,以解决这个问题。
  • 另一种输出格式 - XStream 支持其它的输出格式,如 JSON。

常见用途

传输, 持久化, 配置, 单元测试

下载

官网下载
http://x-stream.github.io/download.html

或者 maven 引入

<!-- https://mvnrepository.com/artifact/com.thoughtworks.xstream/xstream -->
<dependency>
    <groupId>com.thoughtworks.xstream</groupId>
    <artifactId>xstream</artifactId>
    <version>1.4.11.1</version>
</dependency>

记住以下步骤

  1. 创建 XStream 对象
XStream xstream = new XStream(new StaxDriver());
  1. 序列化对象到 XML
// Object to XML Conversion
String xml = xstream.toXML(student);
  1. 反序列化 XML 获得对象。
// XML to Object Conversion
Student student = (Student) xstream.fromXML(xml);

类混叠

用来创建一个类的 XML 完全限定名称的别名

XStream.alias(String name, Class type)

例如 xstream.alias("student", Student.class);

字段混叠

字段混叠用于创建以 XML 字段的别名。

XStream.aliasField(String alias, Class definedIn, String fieldName)

例如 xstream.aliasField("name", Student.class, "studentName");

隐式集合混叠

使用的集合是表示在 XML 无需显示根。例如,在我们的例子中,我们需要一个接一个,但不是在根节点来显示每一个节点。让我们再次修改例子,下面的代码添加到它。

XStream.addImplicitCollection(Class ownerType, String fieldName)

例如 xstream.addImplicitCollection(Student.class, "notes");

属性混叠

属性混叠用于创建一个成员变量作为 XML 属性序列化。

xstream.useAttributeFor(Student.class, "studentName");
xstream.aliasField("name", Student.class, "studentName");

包混叠

包装混叠用于创建一个类XML的完全限定名称的别名到一个新的限定名称。

xstream.aliasPackage("my.company.xstream", "com.yiibai.xstream");

XStream 注解

XStream 支持使用注解做同样的任务。在前面的章节中,我们已经看到了下面的代码配置。

// 创建一个类的XML完全限定名称的别名
xstream.alias("student", Student.class);
xstream.alias("note", Note.class);

// 使用的集合是表示在XML无需显示根
xstream.addImplicitCollection(Student.class, "notes");

// 成员变量作为XML属性
xstream.useAttributeFor(Student.class, "studentName");
// 用于创建以XML字段的别名
xstream.aliasField("name", Student.class, "studentName");
@XStreamAlias("student")    //define class level alias
class Student {
    @XStreamAlias("name")   //define field level alias
    @XStreamAsAttribute     //define field as attribute
    private String studentName;

    @XStreamImplicit        //define list as an implicit collection
    private List<Note> notes = new ArrayList<Note>();

为了告诉 XStream 框架来处理注释,需要XML序列化之前添加下面的命令。
xstream.processAnnotations(Student.class);
或者
xstream.autodetectAnnotations(true);

XStream高级

XStream对象流

XStream 提供 java.io.ObjectInputStream 和 java.io.ObjectOutputStream 替代实现,使对象流可以被序列化或 XML序列化。当大对象集要被处理,保持在存储器中的一个对象,这是特别有用的。

语法 : createObjectOutputStream()

ObjectOutputStream objectOutputStream = xstream.createObjectOutputStream(new FileOutputStream("test.txt"));

语法 :createObjectInputStream()

ObjectInputStream objectInputStream = xstream.createObjectInputStream(new FileInputStream("test.txt"));

XStream 自定义转换器

XStream 允许从无到有写入转换器,这样开发人员可以编写一个完全新的实现,如何对象序列化到 XML,反之亦然。 转换器接口提供了三种方法。

  • canConvert - 检查支持的对象类型的序列化。
  • marshal - 序列化对象到XML。
  • unmarshal - 从XML对象反序列化
package xx.yy;

import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.converters.Converter;
import com.thoughtworks.xstream.converters.MarshallingContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
import com.thoughtworks.xstream.io.xml.StaxDriver;

public class XStreamTester {
    public static void main(String args[]) {
        XStreamTester tester = new XStreamTester();
        XStream xstream = new XStream(new StaxDriver());
        Student student = tester.getStudentDetails();
        xstream.autodetectAnnotations(true);
        xstream.registerConverter(new StudentConverter()); // 注册转换器
        // Object to XML Conversion
        String xml = xstream.toXML(student);
        System.out.println(xml);
    }

    private Student getStudentDetails() {
        Student student = new Student("Mahesh", "Parashar");
        return student;
    }

}

@XStreamAlias("student")
class Student {

    @XStreamAlias("name")
    private Name studentName;

    public Student(String firstName, String lastName) {
        this.studentName = new Name(firstName, lastName);
    }

    public Name getName() {
        return studentName;
    }
}

class Name {
    private String firstName;
    private String lastName;

    public Name(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }

    public String getFirstName() {
        return firstName;
    }

    public String getLastName() {
        return lastName;
    }
}

class StudentConverter implements Converter {

    public void marshal(Object value, HierarchicalStreamWriter writer,
            MarshallingContext context) {
        Student student = (Student) value;
        writer.startNode("name");
        writer.setValue(student.getName().getFirstName() + ","
                + student.getName().getLastName());
        writer.endNode();
    }

    public Object unmarshal(HierarchicalStreamReader reader,
            UnmarshallingContext context) {
        reader.moveDown();
        String[] nameparts = reader.getValue().split(",");
        Student student = new Student(nameparts[0], nameparts[1]);
        reader.moveUp();
        return student;
    }

    public boolean canConvert(Class object) {
        return object.equals(Student.class);
    }
}

XStream 编写 JSON

XStream 支持 JSON 通过初始化 XStream对象适当的驱动程序。 XStream 目前支持 JettisonMappedXmlDriver 和JsonHierarchicalStreamDriver。 现在,让我们使用 XStream 处理 JSON的代码测试。

package xxx.yyy;

import java.io.Writer;

import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
import com.thoughtworks.xstream.io.json.JsonHierarchicalStreamDriver;
import com.thoughtworks.xstream.io.json.JsonWriter;

public class XStreamTester {
    public static void main(String args[]) {
        XStream xstream = new XStream(new JsonHierarchicalStreamDriver() {
            public HierarchicalStreamWriter createWriter(Writer writer) {
                return new JsonWriter(writer, JsonWriter.DROP_ROOT_MODE);
            }
        });

        Student student = new Student("Mahesh", "Parashar");
        xstream.setMode(XStream.NO_REFERENCES);
        xstream.alias("student", Student.class);
        System.out.println(xstream.toXML(student));
    }
}

@XStreamAlias("student")
class Student {

    public String firstName;
    public String lastName;

    public Student(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }

    public String toString() {
        return "Student [ firstName: " + firstName + ", lastName: " + lastName
                + " ]";
    }
}

XStream 的输入输出

对象转 xml

使用toXML() 方法来获取对象的XML字符串表示。

//Object to XML Conversion
String xml = xstream.toXML(student);

示例

        XStream xs = new XStream(new DomDriver());
        System.out.println(xs.toXML(new Object()));
        // <object/>

        System.out.println(xs.toXML(new Integer(21)));
        // <int>21</int>

        System.out.println(xs.toXML("Hello"));
        // <string>Hello</string>

        List<String> list = new ArrayList<String>();
        for (int i = 0; i < 3; i++) {
            list.add(String.valueOf(i));
        }
        System.out.println(xs.toXML(list));
        /*<list>
          <string>0</string>
          <string>1</string>
          <string>2</string>
        </list>*/

        Map<String, String> map = new HashMap<String, String>();
        map.put("aa", "AA");
        map.put("bb", "BB");
        System.out.println(xs.toXML(map));
        /*
        <map>
          <entry>
            <string>aa</string>
            <string>AA</string>
          </entry>
          <entry>
            <string>bb</string>
            <string>BB</string>
          </entry>
        </map>
        */

        List<Map<String, String>> mapList = new ArrayList<Map<String, String>>(1);
        mapList.add(map);
        System.out.println(xs.toXML(mapList));
        /*
        <list>
          <map>
            <entry>
              <string>aa</string>
              <string>AA</string>
            </entry>
            <entry>
              <string>bb</string>
              <string>BB</string>
            </entry>
          </map>
        </list>
        */

对象转 xml 的注意事项

  1. toXML的时候, 如果字段
  • String ppp = "ttt"; 表现形式为 <ppp>ttt</ppp>
  • String ppp = ""; 表现形式为 <ppp></ppp>
  • String ppp = null; 表现形式为没有该行

若使用了注解, 需要xstream对象上加上
xstream.autodetectAnnotations(true);
或者xstream.processAnnotations(Student.class);

  1. xstream对转 int 的处理, 如果标签内非数字,则直接抛出java.lang.NumberFormatException, 所以得根据使用情况, 否则得写成string类型,之后再自行处理。

xml 转实体

使用 fromXML()方法来从XML对象。

//XML to Object Conversion
Student student1 = (Student)xstream.fromXML(xml);

发现对 <list> 标签会处理为 ArrayList, <map> 会被处理为HashMap

示例

<list>
    <map>
        <entry>
            <string>amnt</string>
            <string>22255</string>
        </entry>
        <entry>
            <string>riskCode</string>
            <string>xxx727xxx</string>
        </entry>
        <entry>
            <string>riskName</string>
            <string>727 附加住院费用B款医疗保险</string>
        </entry>
        <entry>
            <string>prem</string>
            <string>538.57</string>
        </entry>
    </map>
    <map>
        <entry>
            <string>mult</string>
            <string>2</string>
        </entry>
        <entry>
            <string>amnt</string>
            <string>18000</string>
        </entry>
        <entry>
            <string>riskCode</string>
            <string>xxx718xxxx</string>
        </entry>
        <entry>
            <string>riskName</string>
            <string>718 住院补贴医疗保险</string>
        </entry>
        <entry>
            <string>prem</string>
            <string>198</string>
        </entry>
    </map>
</list>

xml 文本 转 对象注意事项

  1. 需要配置忽略xml多余的元素, 否则会抛异常
    mXs.ignoreUnknownElements();

注意:XStream老版本没有这个函数(1.4.5以上版本才有此方法)

参考

XStream 教程™
https://www.yiibai.com/xstream

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352