简单实体类和xml文件的相互转换
最近写一个题目,要求将一组员工实体类转换成xml文件,或将xml文件转换成一组实体类。题目不难,但写完感觉可以利用泛型和反射将任意一个实体类和xml文件进行转换。于是今天下午立马动手
试了下,做了个简单的模型,可以将简单的实体类和xml文件进行相互转换,但对实体类的属性类型有限制,目前只支持String, Integer, Double三种类型。但是后面可以扩展。
我的大概思路是这样的,只要能拿到实体类的类型信息,我就能拿到实体类的全部字段名称和类型,拼属性的set和get方法更是简单明了,这时候只需要通过方法的反射,将xml文件的数据读取出来给这个反射即可。
反过来只要给我一个任意对象,我就能通过反射拿到该对象所有字段的值,这时候在写xml文件即可。
具体代码如下:
package com.pcq.entity;
import java.io.*;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.io.OutputFormat;
import org.dom4j.io.SAXReader;
import org.dom4j.io.XMLWriter;
public class XMLAndEntityUtil {
private static Document document = DocumentHelper.createDocument();
/**
* 判断是否是个xml文件,目前类里尚未使用该方法
* @param filePath
* @return
*/
@SuppressWarnings("unused")
private static boolean isXMLFile(String filePath) {
File file = new File(filePath);
if(!file.exists() || filePath.indexOf(".xml") > -1) {
return false;
}
return true;
}
/**
* 将一组对象数据转换成XML文件
* @param list
* @param filePath 存放的文件路径
*/
public static <T> void writeXML(List<T> list, String filePath) {
Class<?> c = list.get(0).getClass();
String root = c.getSimpleName().toLowerCase() + "s";
Element rootEle = document.addElement(root);
for(Object obj : list) {
try {
Element e = writeXml(rootEle, obj);
document.setRootElement(e);
writeXml(document, filePath);
} catch (NoSuchMethodException | SecurityException
| IllegalAccessException | IllegalArgumentException
| InvocationTargetException e) {
e.printStackTrace();
}
}
}
/**
* 通过一个根节点来写对象的xml节点,这个方法不对外开放,主要给writeXML(List<T> list, String filePath)提供服务
* @param root
* @param object
* @return
* @throws NoSuchMethodException
* @throws SecurityException
* @throws IllegalAccessException
* @throws IllegalArgumentException
* @throws InvocationTargetException
*/
private static Element writeXml(Element root, Object object) throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
Class<?> c = object.getClass();
String className = c.getSimpleName().toLowerCase();
Element ele = root.addElement(className);
Field[] fields = c.getDeclaredFields();
for(Field f : fields) {
String fieldName = f.getName();
String param = fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);
Element fieldElement = ele.addElement(fieldName);
Method m = c.getMethod("get" + param, null);
String s = "";
if(m.invoke(object, null) != null) {
s = m.invoke(object, null).toString();
}
fieldElement.setText(s);
}
return root;
}
/**
* 默认使用utf-8
* @param c
* @param filePath
* @return
* @throws UnsupportedEncodingException
* @throws FileNotFoundException
*/
public static <T> List<T> getEntitys(Class<T> c, String filePath) throws UnsupportedEncodingException, FileNotFoundException {
return getEntitys(c, filePath, "utf-8");
}
/**
* 将一个xml文件转变成实体类
* @param c
* @param filePath
* @return
* @throws FileNotFoundException
* @throws UnsupportedEncodingException
*/
public static <T> List<T> getEntitys(Class<T> c, String filePath, String encoding) throws UnsupportedEncodingException, FileNotFoundException {
File file = new File(filePath);
String labelName = c.getSimpleName().toLowerCase();
SAXReader reader = new SAXReader();
List<T> list = null;
try {
InputStreamReader in = new InputStreamReader(new FileInputStream(file), encoding);
Document document = reader.read(in);
Element root = document.getRootElement();
List elements = root.elements(labelName);
list = new ArrayList<T>();
for(Iterator<Emp> it = elements.iterator(); it.hasNext();) {
Element e = (Element)it.next();
T t = getEntity(c, e);
list.add(t);
}
} catch (DocumentException e) {
e.printStackTrace();
} catch (InstantiationException e1) {
e1.printStackTrace();
} catch (IllegalAccessException e1) {
e1.printStackTrace();
} catch (NoSuchMethodException e1) {
e1.printStackTrace();
} catch (SecurityException e1) {
e1.printStackTrace();
} catch (IllegalArgumentException e1) {
e1.printStackTrace();
} catch (InvocationTargetException e1) {
e1.printStackTrace();
}
return list;
}
/**
* 将一种类型 和对应的 xml元素节点传进来,返回该类型的对象,该方法不对外开放
* @param c 类类型
* @param ele 元素节点
* @return 该类型的对象
* @throws InstantiationException
* @throws IllegalAccessException
* @throws NoSuchMethodException
* @throws SecurityException
* @throws IllegalArgumentException
* @throws InvocationTargetException
*/
@SuppressWarnings("unchecked")
private static <T> T getEntity(Class<T> c, Element ele) throws InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
Field[] fields = c.getDeclaredFields();
Object object = c.newInstance();//
for(Field f : fields) {
String type = f.getType().toString();//获得字段的类型
String fieldName = f.getName();//获得字段名称
String param = fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);//把字段的第一个字母变成大写
Element e = ele.element(fieldName);
if(type.indexOf("Integer") > -1) {//说明该字段是Integer类型
Integer i = null;
if(e.getTextTrim() != null && !e.getTextTrim().equals("")) {
i = Integer.parseInt(e.getTextTrim());
}
Method m = c.getMethod("set" + param, Integer.class);
m.invoke(object, i);//通过反射给该字段set值
}
if(type.indexOf("Double") > -1) { //说明该字段是Double类型
Double d = null;
if(e.getTextTrim() != null && !e.getTextTrim().equals("")) {
d = Double.parseDouble(e.getTextTrim());
}
Method m = c.getMethod("set" + param, Double.class);
m.invoke(object, d);
}
if(type.indexOf("String") > -1) {//说明该字段是String类型
String s = null;
if(e.getTextTrim() != null && !e.getTextTrim().equals("")) {
s = e.getTextTrim();
}
Method m = c.getMethod("set" + param, String.class);
m.invoke(object, s);
}
}
return (T)object;
}
/**
* 用来写xml文件
* @param doc Document对象
* @param filePath 生成的文件路径
* @param encoding 写xml文件的编码
*/
public static void writeXml(Document doc, String filePath, String encoding) {
XMLWriter writer = null;
OutputFormat format = OutputFormat.createPrettyPrint();
format.setEncoding(encoding);// 指定XML编码
try {
writer = new XMLWriter(new FileWriter(filePath), format);
writer.write(doc);
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* 默认使用utf-8的格式写文件
* @param doc
* @param filePath
*/
public static void writeXml(Document doc, String filePath) {
writeXml(doc, filePath, "utf-8");
}
}
假如有个实体类是:
package com.pcq.entity;
import java.io.Serializable;
public class Emp implements Serializable{
private Integer id;
private String name;
private Integer deptNo;
private Integer age;
private String gender;
private Integer bossId;
private Double salary;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getDeptNo() {
return deptNo;
}
public void setDeptNo(Integer deptNo) {
this.deptNo = deptNo;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
public Integer getBossId() {
return bossId;
}
public void setBossId(Integer bossId) {
this.bossId = bossId;
}
public Double getSalary() {
return salary;
}
public void setSalary(Double salary) {
this.salary = salary;
}
}
那么写出来的xml文件格式如下:
<?xml version="1.0" encoding="utf-8"?>
<emps>
<emp>
<id>1</id>
<name>张三</name>
<deptNo>50</deptNo>
<age>25</age>
<gender>男</gender>
<bossId>6</bossId>
<salary>9000.0</salary>
</emp>
<emp>
<id>2</id>
<name>李四</name>
<deptNo>50</deptNo>
<age>22</age>
<gender>女</gender>
<bossId>6</bossId>
<salary>8000.0</salary>
</emp>
</emps>
假如有个实体类如下:
package com.pcq.entity;
public class Student {
private Integer id;
private String name;
private Integer age;
private String gender;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
}
那么写出来的xml文件如下
<?xml version="1.0" encoding="utf-8"?>
<students>
<student>
<id></id>
<name>pcq</name>
<age>18</age>
<gender>男</gender>
</student>
</students>
读取也必须读这种格式的xml文件,才能转换成实体类,要求是实体类的类类型信息(Class)必须要获得到。
另外这里的实体类的属性类型均是Integer,String,Double,可以看到工具类里只对这三种类型做了判断。而且可以预想的是,如果出现一对多的关系,即一个实体类拥有一组另一个类对象的引用,
那xml和实体类的相互转换要比上述的情况复杂的多。lz表示短时间内甚至长时间内也不一定能做的出来,欢迎同道高人指点。
简单实体类和xml文件的相互转换的更多相关文章
- JAVA Spring 简单的配置和操作 ( 创建实体类, 配置XML文件, 调试 )
< 1 > 实体类 Person package java_spring.modle; /** * 一个实体类( Person ) */ public class Person { pri ...
- NHibernate生成实体类、xml映射文件
最近工作电脑装完win10后,之前使用的codeSmith安装不了,索性自己写一个. 界面比较简单,如下图: 第一行为Oracle数据库的连接字符串.连接成功后,填充表到第4行的下拉列表中. 第二行为 ...
- mybatis根据表逆向自动化生成代码(自动生成实体类、mapper文件、mapper.xml文件)
.personSunflowerP { background: rgba(51, 153, 0, 0.66); border-bottom: 1px solid rgba(0, 102, 0, 1); ...
- 反向生成hibernate实体类和映射文件
工欲善其事,必先利其器.我们可以使用IDE来根据数据库中的表反向生成实体类和映射文件,虽然这些东西手写也并不是难度很大,但是如果存在大量的简单工作需要我们做,也会显得很麻烦. 写在前面 我们反向生成的 ...
- 简单Java类与XML之间的转换
需要的jar包:xmlpull_1_0_5.jar,xstream-1.4.1.jar) 1.工具类XstreamUtil package com.learn.util; import com.tho ...
- Eclipse从数据库逆向生成Hibernate实体类和映射文件(Eclipse插件系列之HibernateTools)
♣下载安装Eclipse插件(HibernateTools) ♣Eclipse连接数据库(Mysql5.7) ♣新建hibernate.properties和hibernate.cfg.xml文件 ♣ ...
- Idea创建Hibernate bean类和.xml文件
Idea通过表结构反向生成Hibernate实体类和映射文件 首先:之前通过Eclipse反向生成Hibernate的实体类,很傻瓜式,基本上不用配置.但是Idea通过表结构反向生成hibernate ...
- JAXB—Java类与XML文件之间转换
JAXB-Java类与XML文件之间转换 简介 JAXB(Java Architecture for XML Binding) 是一个业界的标准,是一项可以根据XML Schema产生 ...
- 利用JAXB实现java实体类和xml互相转换
1.应用场景 在使用WebService实现数据上传下载,数据查询时,可以利用JAXB实现java实体类和xml互相转换 2.Demo 2.1 student.java 实体类,包含list(set同 ...
随机推荐
- Maven学习专题--Maven入门及安装
因为项目需要,新项目需要使用Maven开发,但是组内大部分没有接触过maven.我就毅然承担搭建maven环境的任务了.因为一切重头开始,就想把自己的整个搭建环境.项目创建.框架整合和模块管理整个过程 ...
- Java系列 - 用Java8新特性进行Java开发太爽了
本人博客文章网址:https://www.peretang.com/using-java8s-new-features-to-coding-is-awesome/ 前言 从开始写博客到现在已经过去3个 ...
- Linux 学习记录 四(Bash 和 Shell scirpt)
一.什么是 Shell? 狭义的shell指的是指令列方面的软件,包括基本的Linux操作窗口Bash等,广义的shell则包括 图形接口的软件,因为图形接口其实也可以操作各种驱动程序来呼叫核心进行工 ...
- 大数据Python学习大纲
最近公司在写一个课程<大数据运维实训课>,分为4个部分,linux实训课.Python开发.hadoop基础知识和项目实战.这门课程主要针对刚从学校毕业的学生去应聘时不会像一个小白菜一样被 ...
- UglyNumber - 找“丑数”
uglynumber的定义是只能被1,2,3,5整除的数 规定1是第一个uglynumber:以此类推,1 2 3 4 5 6 8 9 10 12 15 16 18 20 24 25 27 30 32 ...
- Unity strip engine code可能会使程序崩溃
最近正在做新大厅的红包推荐口令快速领金币入口拍卖行之类的功能,同事把我的捕鱼整合到他的项目中时出现了闪退的问题,经排查是因为strip engine code选项. Strip engine code ...
- (转)Memcached 在windows下的java使用
Memcached 在windows下的java使用 研究这个东东主要是为了解决在教务管理中选课系统的大并发情况下数据库频繁读写造成速度慢的问题,但要使用WEB服务器的内存,是不是可靠还需要验证, ...
- (转)通过Net-Speeder为搬瓦工提升网速
为了解决丢包问题,最简单粗暴的方法就是双倍发送,即同一份数据包发送两份.这样的话在服务器带宽充足情况下,丢包率会平方级降低.直接优点是降低丢包率,直接缺点是耗费双倍流量.一些延伸影响是更容易触发快速恢 ...
- tomcat 与 java web中url路径的配置以及使用规则详情(长期更新)
首先我们看一下在myeclipse中建立的java web项目的结构 在这里我们需要注意这个webroot也就是我们在tomcat里的webapp里面的应用 之所以每一个项目都有这个webroot,是 ...
- Android常用布局、文件存储与权限、XML
常用的布局 LinearLayout Android 2.2开始fill_parent改名为match_parent ,从API Level为8开始我们可以直接用match_parent来代替fill ...