序列化与反序列化

  1. 序列化:把Java对象转换为字节序列的过程。
    作用:1. 进行持久化操作,写入硬盘中
  1. 用于网络传输
  1. 反序列化:把字节序列恢复为Java对象的过程。
    而在Java中,如果一个对象想要实现序列化,就必须实现两个接口之一:Serializable 、Externalizable

一、使用XSteam将Java对象序列化为XML

首先需要导入XSteam依赖

<!--XML反序列化xstream-->
        <dependency>
            <groupId>com.thoughtworks.xstream</groupId>
            <artifactId>xstream</artifactId>
            <version>1.4.4</version>
        </dependency>
  1. 最基本使用
    Person实体类——实现Serializable接口,否则无法序列化:
package com.pantech.cloud.message.handler.osccaac.controller;

import lombok.Data;

import java.io.Serializable;

/**
 * @author lkh
 * @date 2020/9/24 18:29
 * @description
 */

@Data
public class Person implements Serializable {
    private String name;
    private int age;
}

序列化方法:

public void test() {
        try {
            Person person = new Person();
            person.setName("张三");
            person.setAge(18);

            // 将employee对象序列化为XML
            XStream xStream = new XStream(new DomDriver());
            String xmlData = xStream.toXML(person);

            log.info(xmlData);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

输出结果:

<com.pantech.cloud.message.handler.osccaac.controller.Person>
  <name>张三</name>
  <age>18</age>
</com.pantech.cloud.message.handler.osccaac.controller.Person>
  1. 序列化结果中实体默认都带有包名,如果不想让其显示类全名,只要给该类起个别名即可
    修改后的序列化方法:
public void test() {
        try {
            Person person = new Person();
            person.setName("张三");
            person.setAge(18);

            // 将employee对象序列化为XML
            XStream xStream = new XStream(new DomDriver());

            // 设置Person类的别名,第一个参数要重命名的名称,第二个参数为实体类
            xStream.alias("person", Person.class);//这里的名称就是XML根节点的名称,可任意命名
            
            String xmlData = xStream.toXML(person);

            log.info(xmlData);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

输出结果:

<person>
  <name>张三</name>
  <age>18</age>
</person>
  1. 可修改序列化生成的标签名——两种方法
//对属性取别名
 xStream.aliasField("姓名", Person.class,"name");
@XStreamAlias("姓名") // 自定义标签
private String name;

输出结果:

<person>
  <姓名>张三</姓名>
  <age>18</age>
</person>
  1. 如果此时实体中有些属性不想要序列化为 XML 节点,就可设置忽略属性
    第一种方法是在实体属性上使用 transient 关键字
@Data
public class Person implements Serializable {
    private String name;

    /**
     * 使用transient关键字,表示该字段不序列化
     */
    private transient int age;
}

第二种方法是使用 xStream.omitField(Person .class,“age”); 第一个参数是 传递的类,第二个参数是想要忽略的属性名

// 设置Person类的别名,第一个参数要重命名的名称,第二个参数为实体类
            xStream.alias("person", Person.class);
            // 忽略实体属性
            xStream.omitField(Person .class,"age");
            String xmlData = xStream.toXML(person);

第三种方法是在属性字段上加上注解

@XStreamOmitField // 忽略该字段
private int age;
  1. 实体中的属性可以是另一实体
    Friend实体:
@Data
public class Friend {
    private String name;
    private int age;
}

Person实体:

@Data
public class Person implements Serializable {
    private String name;
    
    private int age;

    private Friend friend;
}

序列化方法:

public void test() {
        try {
            Friend friend = new Friend();
            friend.setName("李四");
            friend.setAge(18);

            Person person = new Person();
            person.setName("张三");
            person.setAge(18);
            person.setFriend(friend);

            // 将employee对象序列化为XML
            XStream xStream = new XStream(new DomDriver());

            // 设置Person类的别名,第一个参数要重命名的名称,第二个参数为实体类
            xStream.alias("person", Person.class);

            String xmlData = xStream.toXML(person);

            log.info(xmlData);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

输出结果:

<person>
  <name>张三</name>
  <age>18</age>
  <friend>
    <name>李四</name>
    <age>18</age>
  </friend>
</person>
  1. 属性中可以是实体的list
    Person实体:
private List<Friend> friends;

序列化方法:

public void test() {
       
           Friend friendA = new Friend();
           friendA.setName("李四");
           friendA.setAge(18);

           Friend friendB = new Friend();
           friendB.setName("王五");
           friendB.setAge(23);

           List<Friend> friendList = new ArrayList<>();
           friendList.add(friendA);
           friendList.add(friendB);

           Person person = new Person();
           person.setName("张三");
           person.setAge(18);
           person.setFriends(friendList);

           // 将employee对象序列化为XML
           XStream xStream = new XStream(new DomDriver());

           // 设置Person类的别名,第一个参数要重命名的名称,第二个参数为实体类
           xStream.alias("person", Person.class);
           xStream.alias("friend", Friend.class);

           String xmlData = xStream.toXML(person);

           log.info(xmlData);
   }

输出结果:

<person>
  <name>张三</name>
  <age>18</age>
  <friends>
    <friend>
      <name>李四</name>
      <age>18</age>
    </friend>
    <friend>
      <name>王五</name>
      <age>23</age>
    </friend>
  </friends>
</person>
  1. 若想去掉集合标签
xStream.addImplicitCollection(Person.class, "friends"); // 去掉集合标签

或者使用注解:

@XStreamImplicit(itemFieldName="friend") // 去掉集合标签
private List<Friend> friends;

输出结果:

<person>
  <name>张三</name>
  <age>18</age>
  <friend>
    <name>李四</name>
    <age>18</age>
  </friend>
  <friend>
    <name>王五</name>
    <age>23</age>
  </friend>
</person>

二、XML反序列化为Java对象

将XML反序列化为Java对象时,需要注意节点与实体属性的对应
xStream的属性设置与序列化是一样的。
原XML数据:

<person>
  <姓名>张三</姓名>
  <age>18</age>
  <address>解放路</address>
  <friend>
    <name>李四</name>
    <age>18</age>
  </friend>
  <friend>
    <name>李四</name>
    <age>18</age>
  </friend>
  <friend>
    <name>王五</name>
    <age>23</age>
  </friend>
</person>

转换为上面的Person类
转换方法:

// 忽略某些节点,作用是 某些不需要的节点时不会报错,如:<address>标签
        XStream xStream = new XStream() {
            @Override
            protected MapperWrapper wrapMapper(MapperWrapper next){
                return new MapperWrapper(next) {
                    @Override
                    public boolean shouldSerializeMember(Class definedIn, String fieldName){
                        if (definedIn == Object.class){
                            try {
                                return this.realClass(fieldName) != null;
                            } catch (Exception e){
                                return false;
                            }
                        } else {
                            return super.shouldSerializeMember(definedIn, fieldName);
                        }
                    }
                };
            }
        };
        xStream.alias("person", Person.class);// 这里对类的重命名一定是根节点名称
        xStream.alias("friend", Friend.class);// 不重命名会报错

        // 属性重命名
        xStream.aliasField("姓名", Person.class,"name");

        // 去掉集合标签
        xStream.addImplicitCollection(Person.class, "friends");
        Person person = (Person) xStream.fromXML(xmlMessage);

        log.info(person.toString());

输出结果:
Person(name=张三, age=18, friends=[Friend(name=李四, age=18), Friend(name=李四, age=18), Friend(name=王五, age=23)])

总结:

  1. 对类重命名,去除完整包名
    - xStream.alias(“person”, Person.class);
    - @XStreamAlias(“person”) // 直接加在类 上
  2. 修改序列化生成的标签名
  • xStream.aliasField(“姓名”, Person.class,“name”);
  • 在类属性上使用注解:@XStreamAlias(“姓名”) // 自定义标签
  1. 忽略类的某些属性
  • 在类属性上使用 transient 关键字
  • xStream.omitField(Person .class,“age”);
  • 注解@XStreamOmitField // 忽略该字段
  1. 去除集合标签
  • xStream.addImplicitCollection(Person.class, “friends”); // 去掉集合标签
  • @XStreamImplicit(itemFieldName=“friend”) // 去掉集合标签
  1. 反序列化时,若忽略某些节点,则可以使用:
XStream xStream = new XStream() {
            @Override
            protected MapperWrapper wrapMapper(MapperWrapper next){
                return new MapperWrapper(next) {
                    @Override
                    public boolean shouldSerializeMember(Class definedIn, String fieldName){
                        if (definedIn == Object.class){
                            try {
                                return this.realClass(fieldName) != null;
                            } catch (Exception e){
                                return false;
                            }
                        } else {
                            return super.shouldSerializeMember(definedIn, fieldName);
                        }
                    }
                };
            }
        };