• 【开发心得】Jaxb使用珠玑


    前言

        Java操作xml转换成javaBean,或者javaBean转换为xml的方式有很多。常见的有dom4j等工具直接操作dom,或者使用jaxb.

    jaxb介绍:

    JAXB(Java Architecture for XML Binding简称JAXB)允许Java开发人员将Java类映射为XML表示方式。JAXB提供两种主要特性:将一个Java对象序列化为XML,以及反向操作,将XML解析成Java对象。换句话说,JAXB允许以XML格式存储和读取数据,而不需要程序的类结构实现特定的读取XML和保存XML的代码。

    工具类

    1. package com.test.tms.backend.service.utils;
    2. import com.test.tms.backend.service.xml.ParseXmlExceptionHandler;
    3. import com.sun.xml.internal.bind.marshaller.CharacterEscapeHandler;
    4. import com.sun.xml.internal.bind.marshaller.NoEscapeHandler;
    5. import com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl;
    6. import com.sun.xml.internal.bind.v2.runtime.JaxBeanInfo;
    7. import com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallerImpl;
    8. import com.sun.xml.internal.bind.v2.util.XmlFactory;
    9. import lombok.extern.slf4j.Slf4j;
    10. import org.apache.commons.io.IOUtils;
    11. import org.apache.commons.lang3.StringEscapeUtils;
    12. import org.apache.commons.lang3.StringUtils;
    13. import org.xml.sax.InputSource;
    14. import org.xml.sax.SAXException;
    15. import org.xml.sax.XMLReader;
    16. import javax.xml.bind.*;
    17. import javax.xml.parsers.ParserConfigurationException;
    18. import javax.xml.parsers.SAXParserFactory;
    19. import javax.xml.transform.sax.SAXSource;
    20. import java.io.InputStream;
    21. import java.io.StringWriter;
    22. import java.nio.charset.StandardCharsets;
    23. /**
    24. * @Author: martin
    25. * @Date: 2023/8/15 09:44
    26. * @Description:
    27. */
    28. @Slf4j
    29. public class XMLUtils {
    30. /**
    31. * 将String类型的xml转换成对象
    32. */
    33. public static Object convertXmlStrToObject(Class clazz, String xmlStr) {
    34. Object xmlObject = null;
    35. try {
    36. JAXBContext context = JAXBContext.newInstance(clazz);
    37. Unmarshaller unmarshaller = context.createUnmarshaller();
    38. unmarshaller.setEventHandler(new ValidationEventHandler() {
    39. @Override
    40. public boolean handleEvent(ValidationEvent event) {
    41. if (event.getSeverity() == 2) {
    42. Throwable t = event.getLinkedException();
    43. if (t != null) {
    44. log.error(t.getMessage(), t);
    45. }
    46. return true;
    47. }
    48. return true;
    49. }
    50. });
    51. InputStream inputStream = IOUtils.toInputStream(xmlStr, StandardCharsets.UTF_8);
    52. InputSource inputSource = new InputSource(inputStream);
    53. XMLReader reader = getXMLReader(context);
    54. reader.setFeature("http://apache.org/xml/features/continue-after-fatal-error", true);
    55. ParseXmlExceptionHandler errorHandler = new ParseXmlExceptionHandler();
    56. reader.setErrorHandler(errorHandler);
    57. SAXSource source = new SAXSource(reader, inputSource);
    58. xmlObject = ((UnmarshallerImpl) unmarshaller).unmarshal0(source, (JaxBeanInfo) null);
    59. } catch (Exception e) {
    60. log.error(e.getMessage(), e);
    61. }
    62. return xmlObject;
    63. }
    64. public static XMLReader getXMLReader(JAXBContext context) throws JAXBException {
    65. try {
    66. SAXParserFactory parserFactory = XmlFactory.createParserFactory(((JAXBContextImpl) context).disableSecurityProcessing);
    67. parserFactory.setValidating(false);
    68. return parserFactory.newSAXParser().getXMLReader();
    69. } catch (ParserConfigurationException var2) {
    70. throw new JAXBException(var2);
    71. } catch (SAXException var3) {
    72. throw new JAXBException(var3);
    73. }
    74. }
    75. /**
    76. * @param javaBean
    77. * @param <T>
    78. * @return
    79. * @description: 将包含@XmlRootElement的jaxb javaBean转换Xml
    80. */
    81. public static <T> String javaBeanToXmlStr(T javaBean, Boolean noHeadFlag, Boolean keepSpecCharFlag) {
    82. try {
    83. JAXBContext context = JAXBContext.newInstance(javaBean.getClass());
    84. Marshaller marshaller = context.createMarshaller();
    85. marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
    86. // 去掉报文头
    87. if (noHeadFlag) {
    88. marshaller.setProperty(Marshaller.JAXB_FRAGMENT, true);
    89. }
    90. // 禁止转义
    91. if (keepSpecCharFlag) {
    92. CharacterEscapeHandler escapeHandler = NoEscapeHandler.theInstance;
    93. marshaller.setProperty("com.sun.xml.internal.bind.characterEscapeHandler", escapeHandler);
    94. }
    95. StringWriter writer = new StringWriter();
    96. marshaller.marshal(javaBean, writer);
    97. return writer.toString();
    98. } catch (Exception e) {
    99. log.error(e.getMessage(), e);
    100. }
    101. return null;
    102. }
    103. /**
    104. * @param str
    105. * @return
    106. * @description: 将xml字符串进行加密
    107. */
    108. public static String escapeXmlStr(String str) {
    109. if (StringUtils.isNotBlank(str)) {
    110. String escapedData = StringEscapeUtils.escapeXml11(str);
    111. return escapedData;
    112. }
    113. return null;
    114. }
    115. }

    珠玑

    聊点心得:

    1. 关于jaxb实体类的快速生成。

    可以借助jxc或者 jdk自带tools,参考博主另一篇博文

    【开发心得】Java xsd文件转JavaBean-CSDN博客最近又要对接友商老的系统,依然采用http + xml方式的请求,客方提供了xsd,这里提供windows平台两种转换xsd文件为javaBean的方式。https://blog.csdn.net/qq_26834611/article/details/133788946?spm=1001.2014.3001.55012. jaxb生成xml去掉报文头

    参考如下设置:

    1. JAXBContext context = JAXBContext.newInstance(javaBean.getClass());
    2. Marshaller marshaller = context.createMarshaller();
    3. marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
    4. // 去掉报文头
    5. if (noHeadFlag) {
    6. marshaller.setProperty(Marshaller.JAXB_FRAGMENT, true);
    7. }

    3. 控制是否需要转义字符

    jaxb控制字符转义,jaxb不转义字符。如下

    1. // 禁止转义
    2. if (keepSpecCharFlag) {
    3. CharacterEscapeHandler escapeHandler = NoEscapeHandler.theInstance;
    4. marshaller.setProperty("com.sun.xml.internal.bind.characterEscapeHandler", escapeHandler);
    5. }
    6. StringWriter writer = new StringWriter();
    7. marshaller.marshal(javaBean, writer);
    8. return writer.toString();

    有资料显示设置如下参数即可。

    com.sun.xml.bind.characterEscapeHandler

    但是实际上,java1.8 (331) 是

    com.sun.xml.internal.bind.characterEscapeHandler

    暂时没有去探究哪个版本开始变更的,遇到这种情况,直接翻一下源码即可。

    它总共有4个实现类,如果不满足你的需求,可以手动实现并且替换即可。

     4. 单独转义特殊字符。

    实际上jaxb对于特殊字符的转义,默认不包含引号,这里使用StringEscapeUtils.escapeXml11()进行转义。

    1. String escapedData = StringEscapeUtils.escapeXml11(str);
    2. return escapedData;

    总结: 主要在实际开发中遇到了"JAXB嵌套,其中一个字段为String,存储的是xml"的情况。

  • 相关阅读:
    欧拉回路总结
    【java表达式引擎】四、高性能、轻量级的AviatorScript
    Python优化算法02——遗传算法
    华为三层交换机:ACL的基本实验
    亿级万物互联新时代的物联网消息中间件EMQX调研
    Constellation 介绍:Chainlink 黑客马拉松
    FLINK 基于1.15.2的Java开发-Watermark是怎么解决延迟数据唯一正确的生产级解决方案-目前市面上的例子都有问题
    Ubuntu18.04+RTX3060显卡配置pytorch、cuda、cudnn和miniconda
    力扣每日一题49:字母异位词分组
    TIA博途V17中ProDiag功能的使用方法示例(一)PLC数据类型的监控
  • 原文地址:https://blog.csdn.net/qq_26834611/article/details/133890374