跳至主要內容

使用 XML 工具: jaxb

Steven大约 1 分钟xmljavajaxb

介绍 Java 原生 oxm 工具 jaxb 的使用。

JAXB 教程
https://github.com/LawssssCat/blog/tree/master/code/demo-java-xml/n11-digester-usage/test/java/org/example/

XML 文件
<?xml version="1.0" encoding="utf-8" ?>
<my-content>
    <web-site>http://www.example.org</web-site>
    <owner>steven</owner>
    <description>原生 java xml 操作测试消息</description>
    <posts>
        <!-- 我是注释 -->
        <post id="P001">
            <title>文章标题01</title>
            <content>hello world 01!</content>
        </post>
        <post id="P002">
            <title>文章标题02</title>
            <content>hello world 02!</content>
        </post>
    </posts>
</my-content>

package org.example;

import org.example.entity.JAXBMyContent;
import org.example.entity.JAXBPost;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
import utils.IOUtils;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

public class XmlJaxbWriteTest {
    public static File FILE = new File(
            Objects.requireNonNull(XmlJaxbWriteTest.class.getResource("/")).getFile(),
            XmlJaxbWriteTest.class.getName() + ".xml"
    );
    @Test
    void test() throws IOException {
        try {
            JAXBContext jaxbContext = JAXBContext.newInstance(JAXBMyContent.class);
            Marshaller marshaller = jaxbContext.createMarshaller();
            // marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE); // 格式化输出
            marshaller.setProperty(Marshaller.JAXB_ENCODING, StandardCharsets.UTF_8.name());
            marshaller.marshal(buildMyContent(), FILE);
        } catch (JAXBException e) {
            throw new RuntimeException(e);
        }
        Assertions.assertTrue(FILE.exists());
        String xmlContent = IOUtils.readFile(FILE);
        Assertions.assertEquals("<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>" +
                        "<my-content>" +
                        "<description>build by jaxp</description><owner>steven</owner>" +
                        "<posts>" +
                            "<post id=\"P000\">" +
                                "<content>JAXP-POST-CONTENT</content>" +
                                "<title>JAXP-TITLE-0</title>" +
                            "</post>" +
                            "<post id=\"P001\">" +
                                "<content>JAXP-POST-CONTENT</content>" +
                                "<title>JAXP-TITLE-1</title>" +
                            "</post>" +
                        "</posts>" +
                        "<web-site>https://org.example</web-site>" +
                        "</my-content>",
                xmlContent);
        Assertions.assertTrue(xmlContent.contains("my-content"));
        Assertions.assertTrue(xmlContent.contains("web-site"));
        Assertions.assertTrue(xmlContent.contains("<post id=\"P000\">"));
    }

    private JAXBMyContent buildMyContent() {
        // 报错1: [com.sun.istack.internal.SAXException2: 由于类型 "org.example.entity.MyContent" 缺少 @XmlRootElement 注释, 无法将该类型编集为元素]
        // 解决1: 需要 JAXPMyContent 类上添加 @XmlRootElement 注解
        // 报错2: [com.sun.istack.internal.SAXException2: 由于类型 "org.example.entity.JAXPMyContent" 对此上下文是未知的, 无法将该类型编集为元素。]
        // 解决2: 需要 MyContent 类上加 @XmlSeeAlso({JAXPMyContent.class}) 注解
        // MyContent myContent = new MyContent();
        JAXBMyContent myContent = new JAXBMyContent();
        myContent.setWebSite("https://org.example");
        myContent.setOwner("steven");
        myContent.setDescription("build by jaxp");
        List<JAXBPost> posts = new ArrayList<>();
        for (int i = 0; i < 2; i++) {
            JAXBPost post = new JAXBPost();
            post.setId("P00" + i);
            post.setTitle("JAXP-TITLE-"+i);
            post.setContent("JAXP-POST-CONTENT");
            posts.add(post);
        }
        myContent.setPosts(posts);
        return myContent;
    }
}

maven 插件 jaxb2-maven-plugin

jaxb2-maven-plugin 是一个用来代替 xjc 和 schemagen 用来完成 java 类和 schema 文件之间互相转换的 maven 插件。

官方地址:https://www.mojohaus.org/jaxb2-maven-plugin/Documentation/v2.2/open in new window (⭐⭐⭐)
github:https://github.com/mojohaus/jaxb2-maven-pluginopen in new window \

mvn jaxb2:schemagen  —— Java 类生成 schema
mvn jaxb2:xjc        —— schema 文件生成 Java

例子: https://github.com/LawssssCat/blog/tree/master/code/demo-java-maven/plugin-usage-jaxb2-maven-plugin/

依赖和插件配置
<?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 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.example</groupId>
        <artifactId>demo-java-maven</artifactId>
        <version>1.0-SNAPSHOT</version>
    </parent>

    <artifactId>plugin-usage-jaxb2-maven-plugin</artifactId>

    <dependencies>
        <dependency>
            <groupId>jakarta.xml.bind</groupId>
            <artifactId>jakarta.xml.bind-api</artifactId>
        </dependency>
    </dependencies>

    <build>
        <resources>
            <resource>
                <directory>${project.basedir}/src/main/resources/</directory>
            </resource>
        </resources>
        <plugins>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>jaxb2-maven-plugin</artifactId>
                <version>3.2.0</version>
                <executions>
                    <!-- Java to Schema -->
                    <execution>
                        <id>schemagen</id>
                        <goals>
                            <goal>schemagen</goal>
                        </goals>
                        <configuration>
                            <encoding>UTF-8</encoding>
                            <sources>
                                <source>${project.build.sourceDirectory}/org/example/entity/Person.java</source>
                            </sources>
                            <outputDirectory>${project.basedir}/src/main/jaxb2/schemagen/</outputDirectory>
                        </configuration>
                    </execution>
                    <!-- Schema to Java -->
                    <execution>
                        <id>xjc</id>
                        <goals>
                            <goal>xjc</goal>
                        </goals>
                        <configuration>
                            <clearOutputDir>false</clearOutputDir>
                            <packageName>org.example.gen.entity</packageName>
                            <sources>
                                <source>${project.basedir}/src/main/jaxb2/schemagen/schema1.xsd</source>
                            </sources>
                            <outputDirectory>${project.build.sourceDirectory}</outputDirectory>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

</project>

maven 插件 maven-jaxb2-plugin

maven 插件 maven-jaxb2-plugin 实现了基于 xsd 配置文件生成对应的 Java 类的功能。 在配置各种模型,属性很多的时候,生成相关文件非常方便。

Github: https://github.com/highsource/jaxb-tools?tab=readme-ov-file#jaxb-maven-pluginopen in new window

<plugin>
  <groupId>org.jvnet.jaxb2.maven2</groupId>
  <artifactId>maven-jaxb2-plugin</artifactId>
  <version>0.14.0</version>
  <executions>
    <execution>
      <id>generate</id>
      <goals>
        <goal>generate</goal>
      </goals>
      <configuration>
        <schemaDirectory>src/main/resources/tmsad</schemaDirectory>
        <schemaIncludes>
          <include>*.xsd</include>
        </schemaIncludes>
      </configuration>
    </execution>
  </executions>
</plugin>