今天leadr提出需求,原来公司项目中读取解析xml文件的代码效率太低,考虑切换一种xml为数据封装格式与读取方式以提高效率。我这灵机一动spring对bean的依赖注入就是读取xml文件,可以尝试扒一扒spring的源码,来实现一个轻量级的方案。

重构xml文件,向spring的xml文件格式看齐

重构完成的xml文件格式如下:

<?xml version="1.0" encoding="UTF-8"?>

<beans>
<bean name="iaminformation" class="com.whf.readxml.model.IamConfig">
<property name="iamUrl" value="http:192.168.7.154:8080" />
<property name="token" value="abcdefg" />
<property name="iamName" value="中电科" />
<property name="sourceNumber" value="4" />
<property name="timeSpan" value="12" />
</bean> <bean name="plugin" class="com.whf.readxml.model.PluginAttributes">
<property name="name" value="ADTest1" />
<property name="pluginType" value="AdPlugin" />
<property name="code" value="100001" />
<property name="url" value="ldap://192.168.7.241:389" />
<property name="adminPsw" value="1qaz2wsx2017" />
<property name="adminName" value="QUARKDATA\administrator" />
<property name="adDn" value="OU=test,DC=quarkdata,DC=com" />
<property name="securityAuthentication" value="simple" /> <property name="staffFieldMatch">
<bean name="staffFieldMatch" class="com.whf.readxml.model.StaffFieldMatch">
<property name="idField" value="objectGUID" />
<property name="userNameField" value="sAMAccountName" />
<property name="firstNameField" value="givenName" />
<property name="lastNameField" value="sn" />
<property name="displayNameField" value="displayName" />
<property name="phoneNumberField" value="" />
<property name="telField" value="homePhone" />
<property name="emailField" value="email" />
</bean>
</property> <property name="orgFieldMatch">
<bean name="orgFieldMatch" class="com.whf.readxml.model.OrgFieldMatch">
<property name="idField" value="objectGUID" />
<property name="nameField" value="name" />
<property name="displayNameField" value="ou" />
</bean>
</property> <property name="groupFieldMatch">
<bean name="groupFieldMatch" class="com.whf.readxml.model.GroupFieldMatch">
<property name="idFied" value="objectGUID" />
<property name="nameField" value="sAMAccountName" />
<property name="displayNameField" value="displayName" />
<property name="decriptionField" value="info" />
</bean>
</property>
</bean> <bean name="plugin" class="com.whf.readxml.model.PluginAttributes">
<property name="name" value="LDAPTest1" />
<property name="pluginType" value="LdapPlugin" />
<property name="code" value="100002" />
<property name="url" value="ldap://192.168.7.245/" />
<property name="adminPsw" value="test123456" />
<property name="adminName" value="cn=admin,dc=thundersoft,dc=com" />
<property name="adDn" value="dc=thundersoft,dc=com" />
<property name="securityAuthentication" value="simple" />
<property name="staffFieldMatch">
<bean name="staffFieldMatch" class="com.whf.readxml.model.StaffFieldMatch">
<property name="idField" value="gidNumber" />
<property name="userNameField" value="uid" />
<property name="firstNameField" value="givenName" />
<property name="lastNameField" value="sn" />
<property name="displayNameField" value="displayName" />
<property name="phoneNumberField" value="telephoneNumber" />
<property name="telField" value="tel" />
<property name="emailField" value="email" />
</bean>
</property> <property name="orgFieldMatch">
<bean name="orgFieldMatch" class="com.whf.readxml.model.OrgFieldMatch">
<property name="idField" value="dn" />
<property name="nameField" value="ou" />
<property name="displayNameField" value="orgDisplayName" />
</bean>
</property> <property name="groupFieldMatch">
<bean name="groupFieldMatch" class="com.whf.readxml.model.GroupFieldMatch">
<property name="idFied" value="dn" />
<property name="nameField" value="cn" />
<property name="displayNameField" value="groupDisplayName" />
<property name="decriptionField" value="description" />
</bean>
</property>
</bean>
</beans>

看起来很眼熟的有没有,跟spring的配置文件一样哦。

扒一扒spring读取xml文件的源码

手动扒了一下spring读取xml文件的代码,由于spring过于庞大,读取spring的xml方法又按照读取不同的标签被分拆出n多个方法,楼主能力有限就不在这里卖弄了,不过spring从配置文件把bean加载到bean工厂跟楼主下面读取xml文件的方式理论上是一样的。

废话不多说,上代码:

/**
* 模拟spring依赖注入的方式读取xml文件.
* @author whf
* @date Aug 23, 2017
*/
public class XMLBeanFactory { private String xmlName;
private SAXReader reader;
private Document document; /**
* 构造方法.
* @param xmlName xmlName.
*/
public XMLBeanFactory(String xmlName) { // 在构造方法中
try {
this.xmlName = xmlName;
reader = new SAXReader();
document = reader.read(this.getClass().getClassLoader().getResourceAsStream(xmlName));
} catch (DocumentException e) {
e.printStackTrace();
}
} /**
* 获取相同类型的bean.
* @param type bean的class type.
* @return 返回相同类型的bean的list.
* @throws Exception Exception.
*/
public <T> List<T> getBeansOfType(Class<T> type) throws Exception {
List<T> objects = new ArrayList<>();
try {
Element root = document.getRootElement();
List<Element> beans = root.elements();
if (beans.size() > 0) {
for (Element bean : beans) {
if (bean.attributeValue("class").equals(type.getName())) {
T object = null;
String clazz = bean.attributeValue("class");
// 通过反射来创建对象
Class beanClass = Class.forName(clazz);
object = (T) beanClass.newInstance(); List<Element> propertys = bean.elements(); if (propertys.size() > 0) {
for (Element property : propertys) {
String key = property.attributeValue("name");
Field field = beanClass.getDeclaredField(key);
field.setAccessible(true); List<Element> childBean = property.elements(); // 如果property下内嵌bean
if (childBean.size() > 0) {
Object childObject = getBean(key, property);
field.set(object, childObject);
} else {
/*
* 此属性值是一个字符串.这里单独处理int,float类型变量.如果不处理,
* 会将String类型直接赋值给int类型,发生ClassCastException
*/
String value = property.attributeValue("value");
// 需要对类型进行判断
if (field.getType().getName().equals("int")) {
// 整数
int x = Integer.parseInt(value);
field.set(object, x);
continue;
}
if (field.getType().getName().equals("float")) {
// 浮点数
float y = Float.parseFloat(value);
field.set(object, y); // 注意double可以接受float类型
continue;
}
field.set(object, value);// 处理String类型
}
}
}
objects.add(object);
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
return objects;
}
/**
* 获取property内嵌的bean.
* @param name id或者bean的name
* @param root 根节点.
* @return 返回封装完整的bean.
* @throws Exception Exception.
*/
public Object getBean(String name, Element root) throws Exception { Object object = null;
List<Element> beans = root.elements();
if (beans.size() > 0) {
for (Element bean : beans) {
if (bean.attributeValue("name").equals(name)) {
// 如果bean name相同则开始创建对象
String clazz = bean.attributeValue("class");
// 通过反射来创建对象
Class beanClass = Class.forName(clazz);
object = beanClass.newInstance(); List<Element> propertys = bean.elements(); if (propertys.size() > 0) {
for (Element property : propertys) {
String key = property.attributeValue("name");
Field field = beanClass.getDeclaredField(key);
field.setAccessible(true); List<Element> childBean = property.elements(); // 如果property下内嵌bean
if (childBean.size() > 0) {
field.set(object, getBean(key, property));
} if (property.attribute("ref") != null) {
/*
* 此属性的值是一个对象.这里由于直接调用getBean方法赋值给对象,返回的对象一定是Bean参数的对象, 因此强制转换不会出问题
*/
String refid = property.attributeValue("ref");
field.set(object, getBean(refid));
} else {
/*
* 此属性值是一个字符串.这里单独处理int,float类型变量.如果不处理,会将String类型直接赋值给int类型,
* 发生ClassCastException
*/
String value = property.attributeValue("value");
// 需要对类型进行判断
field.set(object, value);// 处理String类型
}
}
} }
}
} return object;
} }

楼主的代码就是实现读取xml文件中相同类型的bean封装到list中返回。具体如何实现,看代码,注释写的很清楚了。

文件读取的效率提升120多倍

代码完成之后,对比之前的读取xml的代码,比之前的效率提升了120。

总结一下

大牛都说要看开源框架的源码,收获会怎样怎样,但是对于大部分向我这样的伪码农去扒源码的时候总是一头雾水,不知所云。然而,当我们真正有需求的时候,开源代码的实现便成了一份巨大的宝藏,带着我们的目的去扒源码,有时候会有事半功倍的效果。楼主亲测有效,源码虽好,可不要贪杯哦。

扒一扒spring,dom4j实现模拟实现读取xml的更多相关文章

  1. Dom4j使用Xpath语法读取xml节点

    我们可以使用Xpath的语法来轻易的读取xml的某个节点[类似于jQuery的选择器]: 使用Xpath语法需要添加新的jaxen-1.1-beta-7.rar 这个jar包 dom4j完整jar包我 ...

  2. 菜鸟学Java(八)——dom4j详解之读取XML文件

    dom4j是一个Java的XML API,类似于jdom,用来读写XML文件的.dom4j是一个非常非常优秀的Java XML API,具有性能优异.功能强大和极端易用使用的特点,同时它也是一个开放源 ...

  3. spring源码系列(十): 读取xml入口类 ClassPathXmlApplicationContext 分析

    环境准备: 使用spring5.1.6版本 1 xml配置文件 <?xml version="1.0" encoding="UTF-8"?> < ...

  4. Spring相关:jdom学习:读取xml文件

    云课堂马士兵的spring2.5课程中提到的 用JDOM读取XML文件需先用org.jdom.input.SAXBuilder对象的build()方法创建Document对象,然后用Document类 ...

  5. DOM4J 读取XML配置文件进行数据库连接

        介绍介绍DOM4J.    据说是非常优秀非常优秀的Java XML  API(Dom4j is an easy to use, open source library for working ...

  6. 扒一扒@Retryable注解,很优雅,有点意思!

    你好呀,我是歪歪. 前几天我 Review 代码的时候发现项目里面有一坨逻辑写的非常的不好,一眼望去简直就是丑陋之极. 我都不知道为什么会有这样的代码存在项目里面,于是我看了一眼提交记录准备叫对应的同 ...

  7. linux2.6.24内核源代码分析(2)——扒一扒网络数据包在链路层的流向路径之一

    在2.6.24内核中链路层接收网络数据包出现了两种方法,第一种是传统方法,利用中断来接收网络数据包,适用于低速设备:第二种是New Api(简称NAPI)方法,利用了中断+轮询的方法来接收网络数据包, ...

  8. linux2.6.24内核源代码分析(1)——扒一扒sk_buff

    最近研究了linux内核的网络子系统上的网络分组的接收与发送的流程,发现这个叫sk_buff的东西无处不在,内核利用了这个结构来管理分组,在各个层中传递这个结构,因此sk_buff可以说是linux内 ...

  9. Spring依赖注入 --- 模拟实现

    Spring依赖注入 --- 模拟实现 面向接口编程,又称面向抽象编程, 数据库如果发生更改,对应的数据访问层也应该改变多写几个实现,需要用谁的时候在service里new谁就可以了面向抽象编程的好处 ...

随机推荐

  1. BZOJ3052:[WC2013]糖果公园

    浅谈莫队:https://www.cnblogs.com/AKMer/p/10374756.html 题目传送门:https://lydsy.com/JudgeOnline/problem.php?i ...

  2. bzoj 2115 [Wc2011] Xor——路径和环的转化

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2115 思路很精妙.好像能应用到很多地方. 发现如果路径上有环,可以通过一些走法达到 异或了那 ...

  3. [转载]python的range()函数用法

    使用python的人都知道range()函数很方便,今天再用到他的时候发现了很多以前看到过但是忘记的细节.这里记录一下range(),复习下list的slide,最后分析一个好玩儿的冒泡程序. 这里记 ...

  4. Day3-Python基础3--函数参数及调用

    一.return返回值 return的两个作用: 1)需要用一个变量来接受程序结束后返回的结果 2)它是作为一个结束符,终止程序运行 def test(): print("我是return前 ...

  5. debug时打到了URLClassLoader.class里面,

    一.解决方法,查看breakpoints,看有没有在这个类里面打断点,有时会系统自动打断电在这个类里面, 二.在设置里面,找到debug,去掉debug的前面几个断电设置.

  6. Spring MVC配置详解(1)

    web.xml的配置 <!-- 配置前端控制器 前端控制器(DispatcherServlet)--> <servlet> <servlet-name>spring ...

  7. Java 的三个注释

    单行注释 // 这是名为 a 的类 class a{ } 多行注释 /* 这是多行注释 可以注释多行 */ class a{ } 文档注释 /** 这是文档注释 可以注释多行 */ class a{ ...

  8. 问题:table 可否实现对角线;结果:用div+css模拟表格对角线

    首先声明: 这只是探讨一种CSS模拟表格对角线的用法,实际在工作中可能觉得这样做有点小题大作,这不是本主题讨论的重点.如果对此深以为然的朋友,请一笑过之... 有时在插入文档时,要用到表格对角线,常见 ...

  9. C语言学习笔记--指针和数组的关系

    1.数组的本质 (1)数组是一段连续的内存空间 (2)数组的空间大小:sizeof(array_type)*array_size; (3)数组名可看做指向数组第一个元素的常量指针 (4)数组声明时编译 ...

  10. Android 使用SWIG生成Jni代码<转>

    http://blog.csdn.net/u010780612/article/details/51066819 感觉也就那样把,我没有编译成功,实用性暂时没看到多大优点...