所有分类
  • 所有分类
  • 未分类

Spring之Jackson–转换XML与Java对象的方法

简介

本文介绍使用Spring自带的Jackson转换XML与Java对象的方法。

XML与Java对象转换的方法有很多,最好的方法是使用Spring自带的XmlMapper,本文介绍此工具的用法。

XmlMapper类是 ObjectMapper类的子类,两者使用方式几乎一致。

依赖

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

<dependency>
    <groupId>com.fasterxml.jackson.dataformat</groupId>
    <artifactId>jackson-dataformat-xml</artifactId>
    <!-- 版本由spring-boot-starter-parent指定-->
</dependency>

jackson-dataformat-xml依赖了jackson-core、jackson-databind、jackson-annotations等,这些都在spring-boot-starter-web中引入了。

整个pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.4.13</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.knife</groupId>
    <artifactId>Demo_XML_SpringBoot</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>Demo_XML_SpringBoot</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>com.fasterxml.jackson.dataformat</groupId>
            <artifactId>jackson-dataformat-xml</artifactId>
        </dependency>

        <dependency>
            <groupId>com.github.xiaoymin</groupId>
            <artifactId>knife4j-spring-boot-starter</artifactId>
            <version>3.0.3</version>
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.24</version>
            <scope>provided</scope>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

工具类

package com.knife.example.common.util;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.MapperFeature;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.deser.std.DateDeserializers;
import com.fasterxml.jackson.databind.ser.std.DateSerializer;
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
import com.fasterxml.jackson.dataformat.xml.ser.ToXmlGenerator;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
import lombok.SneakyThrows;

import java.text.SimpleDateFormat;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.Date;

public class XmlUtil {
    private static final XmlMapper xmlMapper = createXmlMapper();

    private static XmlMapper createXmlMapper() {
        XmlMapper xmlMapper = XmlMapper.builder()
                // // 忽略实体类没有对应属性。如果为 true 会抛出异常
                // .configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES,false)
                // // 忽略null
                // .serializationInclusion(JsonInclude.Include.NON_NULL)
                // // 属性使用 驼峰首字母小写
                // .propertyNamingStrategy(PropertyNamingStrategy.LOWER_CAMEL_CASE)
                // .defaultUseWrapper(false)
                .build();

        // 序列化时加上文件头信息
        xmlMapper.configure(ToXmlGenerator.Feature.WRITE_XML_DECLARATION, true);
        // 美化输出结果(给XML添加缩进)
        xmlMapper.enable(SerializationFeature.INDENT_OUTPUT);

        // 把“忽略重复的模块注册”禁用,否则下面的注册不生效
        xmlMapper.disable(MapperFeature.IGNORE_DUPLICATE_MODULE_REGISTRATIONS);
        xmlMapper.registerModule(configTimeModule());
        // 重新设置为生效,避免被其他地方覆盖
        xmlMapper.enable(MapperFeature.IGNORE_DUPLICATE_MODULE_REGISTRATIONS);

        return xmlMapper;
    }

    private static JavaTimeModule configTimeModule() {
        JavaTimeModule javaTimeModule = new JavaTimeModule();

        String localDateTimeFormat = "yyyy-MM-dd HH:mm:ss";
        String localDateFormat = "yyyy-MM-dd";
        String localTimeFormat = "HH:mm:ss";
        String dateFormat = "yyyy-MM-dd HH:mm:ss";

        // 序列化
        javaTimeModule.addSerializer(
                LocalDateTime.class,
                new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(localDateTimeFormat)));
        javaTimeModule.addSerializer(
                LocalDate.class,
                new LocalDateSerializer(DateTimeFormatter.ofPattern(localDateFormat)));
        javaTimeModule.addSerializer(
                LocalTime.class,
                new LocalTimeSerializer(DateTimeFormatter.ofPattern(localTimeFormat)));
        javaTimeModule.addSerializer(
                Date.class,
                new DateSerializer(false, new SimpleDateFormat(dateFormat)));

        // 反序列化
        javaTimeModule.addDeserializer(
                LocalDateTime.class,
                new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(localDateTimeFormat)));
        javaTimeModule.addDeserializer(
                LocalDate.class,
                new LocalDateDeserializer(DateTimeFormatter.ofPattern(localDateFormat)));
        javaTimeModule.addDeserializer(
                LocalTime.class,
                new LocalTimeDeserializer(DateTimeFormatter.ofPattern(localTimeFormat)));
        javaTimeModule.addDeserializer(Date.class, new DateDeserializers.DateDeserializer(){
            @SneakyThrows
            @Override
            public Date deserialize(JsonParser jsonParser, DeserializationContext dc){
                String text = jsonParser.getText().trim();
                SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
                return sdf.parse(text);
            }
        });

        return javaTimeModule;
    }

    public static String toXml(Object o) {
        String xml = null;
        try {
            xml = xmlMapper.writeValueAsString(o);
        } catch (JsonProcessingException e) {
            throw new RuntimeException(e);
        }
        return xml;
    }

    public static <T> T toObject(String xml, Class<T> cls) {
        T t = null;

        try {
            t = xmlMapper.readValue(xml, cls);
        } catch (JsonProcessingException e) {
            throw new RuntimeException(e);
        }

        return t;
    }
}

测试类

package com.knife.example.business.controller;

import com.knife.example.business.bo.SchoolBO;
import com.knife.example.business.bo.StudentBO;
import com.knife.example.common.util.XmlUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.List;

@Api(tags = "测试XML")
@RestController
@RequestMapping("xml")
public class TestController {

    @ApiOperation("对象转XML")
    @PostMapping("toXmlString")
    public String toXmlString() {
        SchoolBO schoolBO = new SchoolBO();
        schoolBO.setSchoolId(222);
        schoolBO.setSchoolName("ABC中学");
        schoolBO.setSchoolEmail("aa@163.com");
        schoolBO.setSchoolAddress("A省B市");

        List<StudentBO> studentBOList = new ArrayList<>();
        StudentBO studentBO1 = new StudentBO();
        studentBO1.setStudentName("Tony");
        studentBO1.setCreateTime(LocalDateTime.now());

        StudentBO studentBO2 = new StudentBO();
        studentBO2.setStudentName("Peter");
        studentBO2.setCreateTime(LocalDateTime.now().plusSeconds(2));

        studentBOList.add(studentBO1);
        studentBOList.add(studentBO2);

        schoolBO.setStudentBOList(studentBOList);

        return XmlUtil.toXml(schoolBO);
    }

    @ApiOperation("XML转对象")
    @PostMapping("toObject")
    public Object toObject(String xml) {
        return XmlUtil.toObject(xml, SchoolBO.class);
    }

}

实体类

  • @JacksonXmlRootElement:指定生成xml根标签的名字
    • 属性:namespace,localName
  • @JacksonXmlProperty:指定包装标签名,或者指定标签内部属性名
    • 属性:namespace,localName,isAttribute(default:false)
  • @JacksonXmlElementWrapper:可用于List等集合类,用来指定外围标签名
    • 属性:namespace,localName,useWrapping (default:true)
  • @JacksonXmlText:指定当前这个值,没有xml标签包裹。
    • 属性:namespace,localName

顶层实体类

package com.knife.example.business.bo;

import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlElementWrapper;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlProperty;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlRootElement;
import lombok.Data;

import java.util.List;

@Data
@JacksonXmlRootElement(localName = "school")
public class SchoolBO {
    // 可以指定XML标签的名字。若不指定,则为字段名
    @JacksonXmlProperty(localName = "SchoolName")
    private String schoolName;

    // isAttribute 设为true时,该字段做为根标签的属性
    @JacksonXmlProperty(isAttribute = true)
    private Integer schoolId;

    private String schoolEmail;

    private String schoolAddress;

    // 指定外层的XML标签名
    @JacksonXmlElementWrapper(localName = "students")
    // 指定单个元素的XML标签名
    @JacksonXmlProperty(localName = "item")
    private List<StudentBO> studentBOList;
}

子层实体类

package com.knife.example.business.bo;

import lombok.Data;

import java.time.LocalDateTime;

@Data
public class StudentBO {
    private String studentName;

    private LocalDateTime createTime;
}

测试

1.对象转XML

2.XML转对象

使用此XML测试:

<?xml version='1.0' encoding='UTF-8'?>
<school schoolId="222">
  <schoolEmail>aa@163.com</schoolEmail>
  <schoolAddress>A省B市</schoolAddress>
  <SchoolName>ABC中学</SchoolName>
  <students>
    <item>
      <studentName>Tony</studentName>
      <createTime>2023-03-06 19:46:42</createTime>
    </item>
    <item>
      <studentName>Peter</studentName>
      <createTime>2023-03-06 19:46:44</createTime>
    </item>
  </students>
</school>

结果:

0

评论0

请先

显示验证码
没有账号?注册  忘记密码?

社交账号快速登录