单层XML结构转换为对象数组 - Jackson

在使用到XML的项目中,有时候会把子对象数组打平为单层XML,每一个对象都用一个序号表示。 但是这种XML结构在转换为对象的时候是不方便的,没办法去定义一个类似property_$n的属性。本文利用Jackson和自定义注解可以实现单层XML到对象数组的转换

需求说明

  • 假如需要把下面的XML转换为对象(后面定义的Major
<xml>
    <name>计算机科学</name>
    <year>4</year>
    <name_0>离散数学</name_0>
    <content_0>有点难</content_0>
    <hours_0>64</hours_0>
    <name_1>操作系统</name_1>
    <content_1>计算机真奇妙</content_1>
    <hours_1>48</hours_1>
</xml>

上面的XML中,有两个子结构(name, content, hours),因为是单层结构所以都以序号结尾。 这种格式的XML,没办法定义一个完整的对象,再使用Jackson来直接转换。

  • 目标对象

Major对象有一个Subject数组

/**
 * 课程.
 * @author tenmao
 * @since 2019/12/9
 */
@Data
public class Subject {
    private String name;
    private String content;
    private Integer hours;
}

/**
 * 专业.
 * @author tenmao
 * @since 2019/12/9
 */
@Data
public class Major {
    private String name;
    private Integer years;
    @SingleDeckXml
    private List<Subject> subjectList;
}

//专业有多门课程
  • 希望转换后的对象如下(单层的XML结构转换为对象数组了)
Major(name=计算机科学, years=4, subjectList=[Subject(name=离散数学, content=有点难, hours=64), Subject(name=操作系统, content=计算机真奇妙, hours=48)])

转换工具

  • 注解SingleDeckXml
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
//Jackson内置注解,表明这里还有Jackson其他注解,需要被支持
@JacksonAnnotationsInside
//为了防止冲突,使用SingleDeckXml的注解的属性就不会被Jackson转换
@JsonIgnore
public @interface SingleDeckXml {
}
  • 转换实现XmlUtil
public class XmlUtil {
    /**
     * Jackson转换XML到对象时,支持把单级结构转换为子数据List.
     *
     * @param singleDeckXml 单层XML
     * @param resultClass   对象
     * @param xmlMapper     转换使用的Mapper
     * @param <T>           对象类型
     * @return 转换后的对象
     */
    public static <T> T readSingleDeck(String singleDeckXml, Class<T> resultClass, XmlMapper xmlMapper) {
        try {
            final T refundResult = xmlMapper.readValue(singleDeckXml, resultClass);

            //获取被压缩的对象
            Set<Field> compressedField = new HashSet<>();
            for (Field declaredField : resultClass.getDeclaredFields()) {
                SingleDeckXml annotation = declaredField.getAnnotation(SingleDeckXml.class);
                if (annotation == null) {
                    continue;
                }
                //暂时只支持List
                if (declaredField.getType() != List.class) {
                    throw new RuntimeException(SingleDeckXml.class.toString() + " can only use on List.class");
                }
                compressedField.add(declaredField);
            }


            //获取所有属性值
            TreeMap<String, String> allPropertyValueMap = xmlMapper.readValue(singleDeckXml, new TypeReference<TreeMap<String, String>>() {
            });


            //逐个处理被压缩的对象
            for (Field field : compressedField) {
                List<Object> compressedObjects = new ArrayList<>();
                for (int i = 0; i < Integer.MAX_VALUE; i++) {
                    //构造被压缩的对象
                    boolean hasValue = false;
                    //关键:获取List的实际类型
                    Class<?> klass = (Class<?>) ((ParameterizedType) field.getGenericType()).getActualTypeArguments()[0];
                    final Object compressedObj = klass.newInstance();

                    //设置对象的属性值
                    for (Field declaredField : compressedObj.getClass().getDeclaredFields()) {
                        String propertyName = xmlMapper.getSerializationConfig().getPropertyNamingStrategy().nameForField(null, null, declaredField.getName()) + "_" + i;
                        String value = allPropertyValueMap.get(propertyName);
                        if (value == null) {
                            break;
                        } else {
                            declaredField.setAccessible(true);
                            declaredField.set(compressedObj, toObject(declaredField.getType(), value));
                            hasValue = true;
                        }
                    }
                    //没有匹配到值(说明已经没有更多的被压缩对象了)
                    if (!hasValue) {
                        break;
                    } else {
                        compressedObjects.add(compressedObj);
                    }
                }

                //返回被压缩对象
                field.setAccessible(true);
                field.set(refundResult, compressedObjects);

            }
            return refundResult;
        } catch (JsonProcessingException | IllegalAccessException | InstantiationException e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 把String类型转换为目标类型.
     *
     * @param clazz 目标类型
     * @param value 被转换的值
     * @return 转换后的值
     */
    private static Object toObject(Class clazz, String value) {
        if (Boolean.class == clazz || boolean.class == clazz) {
            return Boolean.parseBoolean(value);
        }
        if (Byte.class == clazz || byte.class == clazz) {
            return Byte.parseByte(value);
        }
        if (Short.class == clazz || short.class == clazz) {
            return Short.parseShort(value);
        }
        if (Integer.class == clazz || int.class == clazz) {
            return Integer.parseInt(value);
        }
        if (Long.class == clazz || long.class == clazz) {
            return Long.parseLong(value);
        }
        if (Float.class == clazz || float.class == clazz) {
            return Float.parseFloat(value);
        }
        if (Double.class == clazz || double.class == clazz) {
            return Double.parseDouble(value);
        }
        return value;
    }
}

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269