侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

简单实体类和xml文件的相互转换方法

技术  /  管理员 发布于 7年前   135

最近写一个题目,要求将一组员工实体类转换成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文件的相互转换方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。


  • 上一条:
    Ajax请求发送成功但不进success的解决方法
    下一条:
    基于apicloudAJAX请求代码合集
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 2024.07.09日OpenAI将终止对中国等国家和地区API服务(0个评论)
    • 2024/6/9最新免费公益节点SSR/V2ray/Shadowrocket/Clash节点分享|科学上网|免费梯子(0个评论)
    • 国外服务器实现api.openai.com反代nginx配置(0个评论)
    • 2024/4/28最新免费公益节点SSR/V2ray/Shadowrocket/Clash节点分享|科学上网|免费梯子(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-07
    • 2017-08
    • 2017-09
    • 2018-01
    • 2018-07
    • 2018-08
    • 2018-09
    • 2018-12
    • 2019-01
    • 2019-02
    • 2019-03
    • 2019-04
    • 2019-05
    • 2019-06
    • 2019-07
    • 2019-08
    • 2019-09
    • 2019-10
    • 2019-11
    • 2019-12
    • 2020-01
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2020-10
    • 2020-11
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-03
    • 2022-04
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-12
    • 2024-02
    • 2024-04
    • 2024-05
    • 2024-06
    • 2025-02
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客