JAXB实例入门

阅读更多
JAXB(Java Architecture for XML Binding),JDK标准规范,Java对象和XML之间的转换,和SAX/DOM不同的是无需关注XML解析细节。
  • Marshalling – 把Java对象转换成XML
  • Unmarshalling – 把XML转换成Java对象
版本:
* JDK1.6开始提供JAXB 2.0(建议使用最新版JDK)
* 低于JDK1.6的版本需要下载jaxb-api.jar和jaxb-impl.jar
* JAXB还有JDK以外的很多第三方实现,比如:EclipseLink MOXy

以下代码都去掉了Bean的Setter/Getter,具体可执行的完整工程代码,可从附件下载。

(1) 基本

Java对象 -> XML
@XmlRootElement
public class Sample {
	private int id;
	private String name;
	private boolean flag;
}

		Sample s = new Sample();
		s.setId(101);
		s.setName("basicObj2XML");
		s.setFlag(false);
		
		StringWriter w = new StringWriter();
		// JAXB是JDK封装好的解析器
		JAXB.marshal(s, new StreamResult(w)); // 输出到字符串
		System.out.println(w.toString());

引用


    false
    101
    basicObj2XML



转换后的XML内容可以直接输出到其他流(比如:控制台、文件等)
		Sample s = new Sample();
		s.setId(102);
		s.setName("basicObj2Output");
		s.setFlag(false);

		JAXB.marshal(s, System.out); // 输出到控制台
		JAXB.marshal(s, new File("d:\\jaxb_sample.xml")); // 输出到文件


XML -> Java对象
		String xml = "" 
					+ "" 
					+ "  103" 
					+ "  basicXML2Obj" 
					+ "  false" 
					+ "";
		
		StringReader r = new StringReader(xml);
		Sample ss = JAXB.unmarshal(r, Sample.class);
		System.out.println(ss.toString());


(2) 通过JAXBContext创建解析器

创建Marshaller
		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Marshaller marshaller = jaxbContext.createMarshaller();
		marshaller.marshal(s, System.out);


创建Unmarshaller
		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
		StringReader r = new StringReader(xml);
		Sample sample = unmarshaller.unmarshal(new StreamSource(r), Sample.class).getValue();
		System.out.println(sample.toString());


设置Marshaller属性
marshaller.setProperty();
		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Marshaller marshaller = jaxbContext.createMarshaller();
		// 设置格式化后输出
		marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
		// 设置字符集
		marshaller.setProperty(Marshaller.JAXB_ENCODING, Charset.defaultCharset().name());
		marshaller.marshal(s, System.out);


(3) 采用EclipseLink MOXy作为JAXB provider
1)下载导入eclipselink.jar
2)在POJO的包下新建jaxb.properties
引用
#Sun JAXB
javax.xml.bind.context.factory=com.sun.xml.internal.bind.v2.ContextFactory
#Eclipse MOXy
#javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory

*** SUN的JAXB实现会在xml中添加属性standalone="yes",而EclipseLink MOXy就不会。

(4) 监听器

marshaller.setListener();

Marshaller监听器
public class MarshallListener extends Marshaller.Listener {

	public void beforeMarshal(Object source) {
		System.out.println("BEFORE_MARSHAL fired");
	}

	public void afterMarshal(Object source) {
		System.out.println("AFTER_MARSHAL fired");
	}

}

		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Marshaller marshaller = jaxbContext.createMarshaller();
		// 设置监听器(!!!Sun JAXB实现有Bug会被执行两次,而Eclipse MOXy能正确执行一次!!!)
		marshaller.setListener(new MarshallListener());
		marshaller.marshal(s, System.out);


Unmarshaller监听器
public class UnmarshallListener extends Unmarshaller.Listener {

	@Override
	public void beforeUnmarshal(Object target, Object parent) {
		System.out.println("BEFORE_UNMARSHAL fired");
	}

	@Override
	public void afterUnmarshal(Object target, Object parent) {
		System.out.println("AFTER_UNMARSHAL fired");
	}

}

		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
		// 设置监听器
		unmarshaller.setListener(new UnmarshallListener());
		StringReader r = new StringReader(xml);
		Sample sample = unmarshaller.unmarshal(new StreamSource(r), Sample.class).getValue();
		System.out.println(sample.toString());


(5) 注解

设置内部标签顺序(默认是按标签字母排序)
@XmlType(propOrder={"id", "name", "flag"})
@XmlRootElement
@XmlType(propOrder={"id", "name", "flag"})
public class SampleOrder {
	private int id;
	private String name;
	private boolean flag;
}

引用


    501
    marshallerTagOrder
    false


把字段值作为属性(默认都是标签)
@XmlAttribute
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleAttribute {
	@XmlAttribute
	private int id;
	private String name;
	private boolean flag;
}

引用


    SampleAttribute
    false


不输出某个字段值
@XmlTransient
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleTransient {
	private int id;
	private String name;
	@XmlTransient
	private boolean flag;
}

引用


    503
    SampleTransient


空值处理

@XmlElement(nillable = true)

		SampleEmpty s = new SampleEmpty();
		s.setId(504);
		s.setName("SampleEmpty");
		s.setFlag(false);
		s.setName1(null);
		s.setName2("");
		s.setName3(null);

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleEmpty {
	private int id;
	private String name;
	private boolean flag;
	private String name1;
	private String name2;
	@XmlElement(nillable = true)
	private String name3;
}

引用


    504
    SampleEmpty
    false
   
   


变更输出标签属性名称

@XmlRootElement(name="foobar-tag")
@XmlAttribute(name="foobar-id")
@XmlElement(name="foobar-value")


@XmlRootElement(name="foobar-tag")
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleName {
	@XmlAttribute(name="foobar-id")
	private int id;
	@XmlElement(name="foobar-value")
	private String name;
	private boolean flag;
}

引用


    SampleName
    false


Bean类型
@XmlRootElement
public class SampleBeanNested {
	private int id;
	private String name;
	private boolean flag;
	private People people;
}

public class People {
	private int id;
	private String name;
}

引用


    false
    506
    SampleBeanNested
   
        100
        People
   



List类型

【字符串】默认

@XmlRootElement
public class SampleStringList1 {
	private int id;
	private String name;
	private boolean flag;
	private List emailAddresses;
}

引用


    [email protected]
    [email protected]
    [email protected]
    false
    507
    SampleStringList1


@XmlElementWrapper

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleStringList2 {
	private int id;
	private String name;
	private boolean flag;
	@XmlElementWrapper(name="email-addresses")
	@XmlElement(name="email-address")
	private List emailAddresses;
}

引用


    508
    SampleStringList2
    false
   
        [email protected]
        [email protected]
        [email protected]
   



@XmlList

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleStringList3 {
	private int id;
	private String name;
	private boolean flag;
	@XmlList
	private List emailAddresses;
}

引用


    509
    SampleStringList3
    false
    [email protected] [email protected] [email protected]


【Bean】默认

@XmlRootElement
public class SampleBeanList1 {
	private int id;
	private String name;
	private boolean flag;
	private List peoples;
}

引用


    false
    510
    SampleBeanList1
   
        100
        People1
   

   
        200
        People2
   

   
        300
        People3
   



@XmlElementWrapper

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleBeanList2 {
	private int id;
	private String name;
	private boolean flag;
	@XmlElementWrapper(name="list")
	@XmlElement(name="people")
	private List peoples;
}

引用


    511
    SampleBeanList2
    false
   
       
            100
            People1
       

       
            200
            People2
       

       
            300
            People3
       

   



@XmlElements

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleBeanList3 {
	private int id;
	private String name;
	private boolean flag;
	@XmlElementWrapper(name="list")
	@XmlElements( { 
    	@XmlElement(name = "man", type = Man.class),
		@XmlElement(name = "woman", type = Woman.class) 
	})
	private List peoples;
}

引用


    512
    SampleBeanLis3
    false
   
       
            100
            Man
            Attr1
       

       
            200
            Woman
            Attr2
       

   



枚举类型

@XmlEnum、@XmlEnumValue

@XmlEnum(Integer.class)
@XmlType
public enum BloodType {
    @XmlEnumValue("1") A,
    @XmlEnumValue("2") B,
    @XmlEnumValue("3") O,
    @XmlEnumValue("4") AB
}

@XmlRootElement
public class SampleEnum {
	private int id;
	private String name;
	private boolean flag;
	private BloodType bloodType;
}

引用


    2
    false
    513
    SampleEnum


(6) 适配器

@XmlJavaTypeAdapter

public class DateAdapter extends XmlAdapter {

	private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd hh:mm:ss");

	public Date unmarshal(String v) throws Exception {
		return dateFormat.parse(v);
	}

	public String marshal(Date v) throws Exception {
		return dateFormat.format(v);
	}

}

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleDateAdapter {
	private int id;
	private String name;
	private boolean flag;
	@XmlJavaTypeAdapter(DateAdapter.class)
	private Date date;
}

引用


    601
    SampleDateAdapter
    false
    2017/06/09 09:52:26


(7) Schema检验

SampleSchema.xsd


     
    
        
            
                
                
            
            
        
     
    



marshaller.setSchema();

@XmlRootElement(name="SampleSchema")
@XmlAccessorType(XmlAccessType.FIELD)
public class SampleSchema {
	@XmlAttribute
	private int id;
	private String name;
	private boolean flag;
}

		SampleSchema s = new SampleSchema();
		s.setId(701);
		s.setName("SampleSchema");
		s.setFlag(false);
		
		ClassLoader classLoader = new Main().getClass().getClassLoader();
        File schemaFile = new File(classLoader.getResource("SampleSchema.xsd").getFile());
		
		SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
		Schema schema = sf.newSchema(schemaFile);

		JAXBContext jaxbContext = JAXBContext.newInstance(SampleSchema.class);
		Marshaller marshaller = jaxbContext.createMarshaller();
		marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
		// 设置Schema
		marshaller.setSchema(schema);
		marshaller.marshal(s, System.out);

引用


    SampleSchema
    false


validator.validate(source);

		SampleSchema s = new SampleSchema();
		s.setId(702);
		s.setName("marshallerValidator");
		s.setFlag(false);

		ClassLoader classLoader = new Main().getClass().getClassLoader();
        File schemaFile = new File(classLoader.getResource("SampleSchema.xsd").getFile());
        
		SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
		Schema schema = sf.newSchema(schemaFile);

		JAXBContext jaxbContext = JAXBContext.newInstance(SampleSchema.class);
		
		// 校验XML
		JAXBSource source = new JAXBSource(jaxbContext, s);
        Validator validator = schema.newValidator();
        validator.setErrorHandler(new CustomValidationErrorHandler());
        validator.validate(source);

		Marshaller marshaller = jaxbContext.createMarshaller();
		marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
		marshaller.marshal(s, System.out);


(8) 捕获事件

marshaller.setEventHandler();

public class CustomEventHandler implements ValidationEventHandler {

	@Override
	public boolean handleEvent(ValidationEvent event) {
		System.out.println(event.getSeverity());

		if (event.getSeverity() == ValidationEvent.ERROR
				|| event.getSeverity() == ValidationEvent.FATAL_ERROR) {
			ValidationEventLocator locator = event.getLocator();
			System.out.println("Line:Col[" + locator.getLineNumber() + ":" + locator.getColumnNumber() + "]");
			throw new RuntimeException(event.getMessage(), event.getLinkedException());
		}
		return true;
	}

}

		JAXBContext jaxbContext = JAXBContext.newInstance(Sample.class);
		Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
		// 设置事件处理器
		unmarshaller.setEventHandler(new CustomEventHandler());
		StringReader r = new StringReader(xml);
		Sample sample = unmarshaller.unmarshal(new StreamSource(r), Sample.class).getValue();
		System.out.println(sample.toString());
  • JAXBSample.zip (8 MB)
  • 下载次数: 0

你可能感兴趣的:(JAXB实例入门)