jaxb xml序列化与Java类的转化

前端之家收集整理的这篇文章主要介绍了jaxb xml序列化与Java类的转化前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

JAXB基本使用

JAXB主要用来实现对象和XML之间的序列化和反序列化,关于JAXB的介绍就不多说了,网上一搜一大把,这里主要总结下基本使用方法和一些注意事项

首先定义两个示例类ClassA,ClassB,用于后续的示例演示

@H_301_17@package cn.lzrabbit;

@H_301_17@public @H_301_17@class ClassA {
    @H_301_17@private @H_301_17@int classAId;
    @H_301_17@private String classAName;

    @H_301_17@private ClassB classB;

    @H_301_17@int getClassAId() {
        @H_301_17@return classAId;
    }

    @H_301_17@void setClassAId(@H_301_17@int classAId) {
        @H_301_17@this.classAId = classAId;
    }

    @H_301_17@public String getClassAName() {
        @H_301_17@return classAName;
    }

    @H_301_17@void setClassAName(String classAName) {
        @H_301_17@this.classAName = classAName;
    }

    @H_301_17@public ClassB getClassB() {
        @H_301_17@return classB;
    }

    @H_301_17@void setClassB(ClassB classB) {
        @H_301_17@this.classB = classB;
    }
}
class ClassB { @H_301_17@int classBId; @H_301_17@private String classBName; @H_301_17@int getClassBId() { @H_301_17@return classBId; } @H_301_17@void setClassBId(@H_301_17@int classBId) { @H_301_17@this.classBId = classBId; } @H_301_17@public String getClassBName() { @H_301_17@return classBName; } @H_301_17@void setClassBName(String classBName) { @H_301_17@this.classBName = classBName; } }

用于序列化的XmlUtil

import java.io.StringReader; @H_301_17@import java.io.StringWriter; @H_301_17@import javax.xml.bind.*; @H_301_17@class XmlUtil { @H_301_17@static String toXML(Object obj) { @H_301_17@try { JAXBContext context = JAXBContext.newInstance(obj.getClass()); Marshaller marshaller = context.createMarshaller(); marshaller.setProperty(Marshaller.JAXB_ENCODING,"UTF-8");// //编码格式 marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT,@H_301_17@true); 是否格式化生成的xml串 marshaller.setProperty(Marshaller.JAXB_FRAGMENT,255); font-size:12px!important">false); 是否省略xm头声明信息 StringWriter writer = @H_301_17@new StringWriter(); marshaller.marshal(obj,writer); @H_301_17@return writer.toString(); } @H_301_17@catch (Exception e) { @H_301_17@throw @H_301_17@new RuntimeException(e); } } @SuppressWarnings("unchecked") @H_301_17@static <T> T fromXML(String xml,Class<T> valueType) { @H_301_17@try { JAXBContext context = JAXBContext.newInstance(valueType); Unmarshaller unmarshaller = context.createUnmarshaller(); @H_301_17@return (T) unmarshaller.unmarshal(@H_301_17@new StringReader(xml)); } @H_301_17@new RuntimeException(e.getMessage()); } } }

调用如下:

class MainRun { /** * @param args */ @H_301_17@static @H_301_17@void main(String[] args) { ClassB classB = @H_301_17@new ClassB(); classB.setClassBId(22); classB.setClassBName("B2"); ClassA classA = @H_301_17@new ClassA(); classA.setClassAId(11); classA.setClassAName("A1"); classA.setClassB(classB); System.out.println(XmlUtil.toXML(classA)); } }

输出结果如下:

@H_301_17@<?xml version="1.0" encoding="UTF-8" standalone="yes"@H_301_17@?>
@H_301_17@<classA@H_301_17@>
    classAId@H_301_17@>11@H_301_17@</classAName@H_301_17@>A1classB@H_301_17@>
        classBId@H_301_17@>22classBName@H_301_17@>B2@H_301_17@>
@H_301_17@>

这里要注意以下几点

  1. 要序列化的类加上@XmlRootElement注解,否则会报错(错误提示很清晰,这里就不贴出来了)
  2. JAXB序列化XML时 默认序列化getter和setter,且getter和setter必须成对出现才会被序列化
  3. 属性名称,默认序列化出来的类和属性名称默认是首字母转换为小写,若需要控制属性名称需要在getter或setter上使用@XmlElement(name="ClassAId") 指定名称,这里要注意的是@XmlElement放置在getter或setter上都行,但只能放一个,也就是说不能同时在getter和setter上使用@XmlElement注解
  4. 如何控制根节点名称
    使用@XmlRootElement指定name属性即可,如@XmlRootElement(name="ClassA")
  5. 怎么添加命名空间
    使用@XmlRootElement(namespace="cn.lzrabbit") 指定namespace属性
  6. 怎么精确控制每个属性名称
    JAXB自动转化为首字母小写会导致不可预料的属性名称出现, 不嫌麻烦的话为每个属性设置@XmlElement(name=""),想省事的话使用Field
  7. 怎么样实现序列化时使用Field字段而不是使用setter和getter
    在要使用的类上面加上@XmlAccessorType(XmlAccessType.FIELD)注解,并指定为XmlAccessType.FIELD,这里强烈推荐使用@XmlAccessorType(XmlAccessType.FIELD)注解,因为这样你可以精确的控制每个元素的名称,而不需要为每个属性去设置@XmlElement(name="")注解,当然也可以在Field上使用@XmlElement注解

下面给出使用了使用如上注解后的代码示例

@XmlRootElement(namespace="cn.lzrabbit")
@XmlAccessorType(XmlAccessType.FIELD)
@H_301_17@int classAId;
    
    @XmlElement(name="ClassAName")
    @H_301_17@return classAId;
    }
    @H_301_17@this.classB = classB;
    }
}

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
@H_301_17@int ClassBId;
    @H_301_17@private String ClassBName;

    @H_301_17@return ClassBId;
    }

    @H_301_17@this.ClassBId = classBId;
    }

    @H_301_17@return ClassBName;
    }

    @H_301_17@this.ClassBName = classBName;
    }
}

输出xml为

ns2:classA xmlns:ns2@H_301_17@="cn.lzrabbit"ClassANameClassBIdClassBNamens2:classA

本篇先写到这里,下一篇写下默认命名空间及自定义命名空间前缀的处理

猜你在找的XML相关文章