XStream双下划线问题解决与CDATA标记同时的方案

简介: <div class="markdown_views"><h2 id="1问题">1、问题</h2><p>在微信开发过程中,需要进行xml格式的数据传输。有些微信接口的xml数据中需要加上CDATA标记,而大部分的xml数据的标签名都带有下划线。注意,微信接口中的数据是有下划线的,是“_”不是“-”,让我很郁闷。</p><h2 id="2使用xstream把java对

1、问题

在微信开发过程中,需要进行xml格式的数据传输。有些微信接口的xml数据中需要加上CDATA标记,而大部分的xml数据的标签名都带有下划线。注意,微信接口中的数据是有下划线的,是“_”不是“-”,让我很郁闷。

2、使用XStream把Java对象转成xml格式的数据

            UnifiedOrder unifiedOrder = new UnifiedOrder();
            unifiedOrder.setAppid("123456");
            unifiedOrder.setAttach("hehedesk");
            unifiedOrder.setBody("hehedesk");
            unifiedOrder.setOpenid("5654675");
            unifiedOrder.setSign("0000000000000000");
            XStream stream = new XStream();
            stream.alias("xml", unifiedOrder.getClass());
            String xml = stream.toXML(unifiedOrder);
            System.out.println(xml);

输出XML为:

<xml>
  <appid>123456</appid>
  <attach>hehedesk</attach>
  <body>hehedesk</body>
  <openid>5654675</openid>
  <sign>0000000000000000</sign>
</xml>

3、为数据加上CDATA标记

修改XStream的实现就可以。

        XStream stream = new XStream(new XppDriver(new NoNameCoder()) {

                @Override
                public HierarchicalStreamWriter createWriter(Writer out) {
                    return new PrettyPrintWriter(out) {
                        // 对所有xml节点的转换都增加CDATA标记
                        boolean cdata = true;

                        @Override
                        @SuppressWarnings("rawtypes")
                        public void startNode(String name, Class clazz) {
                            super.startNode(name, clazz);
                        }

                        @Override
                        protected void writeText(QuickWriter writer, String text) {
                            if (cdata) {
                                writer.write("<![CDATA[");
                                writer.write(text);
                                writer.write("]]>");
                            } else {
                                writer.write(text);
                            }
                        }
                    };
                }
            });

输出XML为:

<xml>
  <appid><![CDATA[123456]]></appid>
  <attach><![CDATA[hehedesk]]></attach>
  <body><![CDATA[hehedesk]]></body>
  <openid><![CDATA[5654675]]></openid>
  <sign><![CDATA[0000000000000000]]></sign>
</xml>

4、当对象属性带下划线时,XStream转换成双下划线

            UnifiedOrder unifiedOrder = new UnifiedOrder();
            unifiedOrder.setAppid("123456");
            unifiedOrder.setAttach("hehedesk");
            unifiedOrder.setBody("hehedesk");
            unifiedOrder.setOpenid("5654675");
            unifiedOrder.setSign("0000000000000000");
            unifiedOrder.setMch_id("123456");
            XStream stream = new XStream();
            stream.alias("xml", unifiedOrder.getClass());
            String xml = stream.toXML(unifiedOrder);

输出XML:

<xml>
  <appid>123456</appid>
  <mch__id>123456</mch__id>
  <attach>hehedesk</attach>
  <body>hehedesk</body>
  <openid>5654675</openid>
  <sign>0000000000000000</sign>
</xml>

注意:这里mch_id的下线线由XStream转成了”__”。

度娘上找到的解决方法:

new DomDriver(null,new XmlFriendlyNameCoder("_-", "_"))

经过测试,XmlFriendlyNameCoder与XppDriver不能同时存在。so,问题来了。如何才能让两者共存呢。

5、双下划线问题解决与CDATA标记同时的方案

双下划线问题的产生是因为XStream默认的转换方式中定义了对特殊字符的转换,代码如下:

//XmlFriendlyNameCoder.encodeName(String name)
for (; i < length; i++ ) {
            char c = name.charAt(i);
            if (c == '$' || c == '_' || c <= 27 || c >= 127) {
                break;
            }
 }

也就是说,我们在转换的过程中,不对特殊字符进行转换就可以了。

XppDriver类中有对字符转换的方法:

    /**
     * Encode the node name into the name of the target format.
     * 
     * @param name the original name
     * @return the name in the target format
     * @since 1.4
     */
    public String encodeNode(String name) {
        return nameCoder.encodeNode(name);
    }

这里可以看到,XppDriver的encodeNode是把节点的名称进行格式化。然后调用nameCoder对象对名字进行编译。

我们在XppDriver的子类中,重写此方法,不再像XppDriver那样调用nameCoder来进行编译,而是直接返回节点名称。

@Override
public String encodeNode(String name) {
    return name;
}

完整代码如下:

            XStream stream = new XStream(new XppDriver(new NoNameCoder()) {

                @Override
                public HierarchicalStreamWriter createWriter(Writer out) {
                    return new PrettyPrintWriter(out) {
                        // 对所有xml节点的转换都增加CDATA标记
                        boolean cdata = true;

                        @Override
                        @SuppressWarnings("rawtypes")
                        public void startNode(String name, Class clazz) {
                            super.startNode(name, clazz);
                        }

                        @Override
                        public String encodeNode(String name) {
                            return name;
                        }


                        @Override
                        protected void writeText(QuickWriter writer, String text) {
                            if (cdata) {
                                writer.write("<![CDATA[");
                                writer.write(text);
                                writer.write("]]>");
                            } else {
                                writer.write(text);
                            }
                        }
                    };
                }
            });

输出XML:

<xml>
  <appid><![CDATA[123456]]></appid>
  <mch_id><![CDATA[1111111]]></mch_id>
  <attach><![CDATA[hehedesk]]></attach>
  <body><![CDATA[hehedesk]]></body>
  <openid><![CDATA[5654675]]></openid>
  <trade_type><![CDATA[JSAPI]]></trade_type>
  <sign><![CDATA[0000000000000000]]></sign>
  <device_info><![CDATA[WEB]]></device_info>
</xml>
目录
相关文章
|
8月前
|
机器学习/深度学习 Java 索引
39、一篇文章弄懂 Java 正则表达式中的量词、贪婪、勉强、独占和 String 的 matches 方法的底层【个人感觉非常值得学习】
39、一篇文章弄懂 Java 正则表达式中的量词、贪婪、勉强、独占和 String 的 matches 方法的底层【个人感觉非常值得学习】
75 0
|
8月前
|
Java 开发者
JDK 21中的记录模式(Record Patterns):简化对象匹配与解构
本文将详细介绍JDK 21中引入的新特性——记录模式(Record Patterns)。记录模式是一种强大的语言特性,它允许开发者在switch表达式中使用简化的语法来匹配和解构记录类型(record types)。本文将解释记录模式的概念、语法、使用场景以及与传统模式匹配的区别,并通过示例代码展示记录模式在实际开发中的应用。
|
8月前
|
XML SQL Java
MyBatis【付诸实践 01】mapper.xml 中特殊符号的 2 种处理方法(CDATA区+替代字符举例)
MyBatis【付诸实践 01】mapper.xml 中特殊符号的 2 种处理方法(CDATA区+替代字符举例)
275 0
|
8月前
|
Oracle Java 关系型数据库
Java【代码分享 06】Lamda表达式将List对象中的Map对象的key全部转化为大写或者小写(去除外层循环:可用于Map对象中的key全部转化为大写或者小写)
Java【代码分享 06】Lamda表达式将List对象中的Map对象的key全部转化为大写或者小写(去除外层循环:可用于Map对象中的key全部转化为大写或者小写)
357 0
|
8月前
|
Java 索引
正则表达式源码分析--三个常用类--分组、捕获、反向引用--String 类中使用正则表达式的代码示例和图
正则表达式源码分析--三个常用类--分组、捕获、反向引用--String 类中使用正则表达式的代码示例和图
89 0
|
JSON 前端开发 Java
springboot中自定义JavaBean返回的json对象属性名称大写变小写问题
springboot中自定义JavaBean返回的json对象属性名称大写变小写问题
227 0
|
JSON PHP 数据格式
php数组学习(2):判断并过滤JSON字符串中不包含指定键名的解决方案
php数组学习(2):判断并过滤JSON字符串中不包含指定键名的解决方案
86 0
|
存储 SQL 缓存
【JavaSE专栏38】Java常用类 String 解析,字符串的花式操作
【JavaSE专栏38】Java常用类 String 解析,字符串的花式操作
112 0
EL表达式学习(一):输出复杂的Bean对象
今天学习的内容是:使用EL表达式输出Person类中的普通属性、数组属性,list集合属性和map集合属性 。
EL表达式学习(一):输出复杂的Bean对象
【java】java反射机制,动态获取对象的属性和对应的参数值,并属性按照字典序排序,Field.setAccessible()方法的说明【可用于微信支付 签名生成】
方法1:通过get()方法获取属性值   package com.sxd.test.controller; public class FirstCa{ private Integer num; private String name; private Boole...
2787 0