热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

结合反射与XML实现Java编程的动态性

反射是Java语言被视为动态或准动态语言的一个关键性质,结合反射和XML会帮助我们更快、更方便地实现一些动态代码,从而解决编程中可能遇到的不确定问题。本

反射是 Java 语言被视为动态或准动态语言的一个关键性质,结合反射和 XML 会帮助我们更快、更方便地实现一些动态代码,从而解决编程中可能遇到的不确定问题。本文将结合反射与 XML 对 Java 编程的动态性进行深入浅出的讨论。在理解本文的思想之后,您可以将其应用到程序中以创建灵活的代码。

0 评论:

张 立达, 实习生

2009 年 12 月 30 日

  • +内容


引言

在现实生活中,经常会发生这种情况。我要去商场买菜来准备晚餐,我不知道买什么,但是进入商场之前我会随手拿一个购物筐来装最终决定要买的物品,这样不管我买什么都可以放入其中,结账之后就可以带回家准备晚饭。在开发程序过程中,也会遇到类似情况。有时我们不能确定类的名称、类有哪些属性以及属性的值等,这些内容只有到运行的时候才能确定。这种情况下,不能将类的名称直接固化,那么怎样才能解决这种问题呢?其实开发程序与现实生活有相似之处。在编程之中,购物筐就好比事先准备好的通用接口,这个接口可以用反射的机制来实现,而购物筐中的菜可以由 XML 文件来描述,这样不管买的菜是什么,我都可以从购物筐中取出进而准备晚餐。

基于上述问题,本文将结合反射与 XML 对 Java 编程的动态性进行深入浅出的讨论。在理解本文的思想之后,您可以将其应用到程序中以创建灵活的代码。

回页首

简要说明

反射是 Java 语言被视为动态或准动态语言的一个关键性质,通过这个机制我们可以在运行时加载、探知、使用编译期间完全未知的类。这个机制允许程序在运行时反射加载一个类或透过 Reflection API 取得任何一个已知名称的类的内部信息,包括其 modifiers(例如 public,static 等等)、superclass(例如 Object)、实现之 interfaces(例如 Cloneable),也包括 fields 和 methods 的所有信息。如果我们利用反射结合 XML 中的一些配置信息,就可以做到运行时加载、探知和装配类,将不可确定的因素在运行时确定化。

如果想了解更多关于 Java 编程动态性的内容,您可以参考 Dennis Sosnoski 关于 Java 编程动态方面 的系列文章。另外,本文将使用 dom4j 对 XML 文件进行存取操作。dom4j 是一个非常非常优秀的 Java XMLAPI,具有性能优异、功能强大和极端易用使用的特点。如果您还没有 dom4j.jar,那么请先 下载 DOM4J 并将其导入到您的工程中。为了支持 xpath,您还需要 下载 jaxen-xxx.jar 并将其导入到工程。

回页首

示例开发

基于上述问题,为了便于理解该解决方案,我们编写简单的示例代码来进行阐述。

对于购物筐,用类 Basket 来代表,其代码如下所示,包括两个操作即将食物放到筐内和从筐中取出食物。

清单 1. Basket 类部分代码

public class Basket {private Food food=null;public void put(Food food){this.food=food;}public Food get(){return this.food;}
}

对于各种食物,我们遵循面向接口编程的思想,按如下类图结构进行编码。

图 1. 各种食物的类图
图 1. 各种食物的类图

例如 Tomato 类的代码如下所示:

清单 2. Tomato 类部分代码

public class Tomato implements Vegetable{private String color = null;public void display(){System.out.print("Hi,I am Tomato!");System.out.print("My color is "+color+".");}public String getColor() {return color;}public void setColor(String color) {this.color = color;}
}

准备好这些基本的类和接口之后,我们编写核心处理代码,这些代码结合反射与 XML 实现 Java 编程的动态性,我们将这些处理放入类 Factory 中。核心代码如下所示:

清单 3. Factory 类部分代码

public class Factory {private String file=null;public Factory(){}public Factory(String file){this.file=file;}public T getBean(){T ret=null;//read informations from foods.xml by dom4jSAXReader saxReader=new SAXReader();Document doc=null;try {doc=saxReader.read(new File(file));} catch (DocumentException e) {e.printStackTrace();}//kind is the value of the "choice" element in foods.xml String kind=doc.selectSingleNode("foods/choice").getText();Element element=(Element)doc.selectSingleNode("foods/"+kind+"/"+kind.substring(0, kind.length()-1));//use hashtable to save all properties of a classHashtable hashtbl=new Hashtable();//read the food name and its corresponding class String name=element.attribute("name").getValue();String className=element.attribute("class").getValue();//read all properperties informations,that is name,type and valueList list1=element.selectNodes("property");Iterator iterator1=list1.iterator();while(iterator1.hasNext()){Element element1=iterator1.next();String propertyName=element1.attribute("name").getValue();String propertyType=element1.attribute("type").getValue();String propertyValue=element1.element("value").getText();//record all the properties and values in a hashtablehashtbl.put(propertyName, propertyType+";"+propertyValue);}//call the method to create a object ret=newInstance(name,className,hashtbl);return ret;}private T newInstance(String name,String className,Hashtable hashtbl){T ret=null;try {//create object by reflectionret=(T)Class.forName(className).newInstance();} catch (Exception e) {e.printStackTrace();}for(String key:hashtbl.keySet()){char [] temp=key.toCharArray();temp[0]=(char)(temp[0]-32);String methodName="set"+new String(temp);String value=hashtbl.get(key).split(";")[1];Class type=null;try {//read property type by reflectiontype=Class.forName(hashtbl.get(key).split(";")[0]);} catch (ClassNotFoundException e) {e.printStackTrace();}try {//get method object by reflectionMethod m=ret.getClass().getMethod(methodName, type);//call method by reflectionm.invoke(ret, value);} catch (Exception e) {e.printStackTrace();}}return ret;}
}

对于购物者,我们只给定两个操作:将选取食物放入购物筐和结账。其代码如下所示:

清单 4. Person 类部分代码

public class Person { public void choose(Basket basket){ Factory factory=new Factory("foods.xml"); Food food=factory.getBean(); basket.put(food); } public Food pay(Basket basket){ return basket.get(); }
}

至此,我们只要在 XML 文件中配置我们想要买的食物即可,具体配置很简单。

清单 5. Food.xml 文件的配置


vegetables red nice >

如果我们决定买西红柿,因为它属于蔬菜,所以设定 choice 元素值为 vegetables, 然后在蔬菜元素下配置西红柿节点即可,如上所示。当然,如果我们想买马铃薯,在蔬菜元素下配置马铃薯节点就可以了。如果我们要买梨,因为它属于水果,所以设定 choice 元素值为 fruits, 然后在水果元素下配置梨节点即可,是不是很简单啊!

好的,一切都准备好了,我们开始购物吧!拿个购物筐、选食物、付款,就可以把食物带回家准备晚餐了!

清单 6. 测试类的部分代码

public class Test {public static void main(String[] args) {Person peoson=new Person();Basket basket=new Basket();peoson.choose(basket);Food food=peoson.pay(basket);food.display();}
}

运行结果如下:

清单 7. 运行结果

Hi,I am Tomato!My color is red.

如果我们设定 choice 元素值为 fruits, 然后在水果元素下配置梨节点,运行结果如下:

清单 8. 运行结果

Hi,I am Pear!My shape is nice.

您一定看出来了,我们没有修改任何代码,只是动态配置了 XML 文件就实现了程序运行的动态性。这是不是一件很酷的事情呢?

回页首

总结

通过该示例开发,我们对结合反射与 XML 实现 Java 编程的动态性有了一个基本认识。我们这里只是为了说明问题而利用了一些简单的反射特性。反射的功能非常强大,值得我们更广泛更深刻地去研究,结合 XML 和反射的这些功能会帮助我们更快,更方便地实现一些 动态代码。

参考资料

  • 参考 The Reflection API,查看关于使用反射的高级教程。
  • 参考 Java 理论和实践:了解泛型,了解如何在代码中使用泛型。
  • 查看 http://sourceforge.net/projects/dom4j/,可以学习 dom4j 的使用细节。
  • 查看 xpath 语法,可以学习很多关于 xpath 使用的细节。
  • 查看 developerWorksJava 技术专区,可以找到很多关于 java 反射的文章。


推荐阅读
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • 开发笔记:spring boot项目打成war包部署到服务器的步骤与注意事项
    本文介绍了将spring boot项目打成war包并部署到服务器的步骤与注意事项。通过本文的学习,读者可以了解到如何将spring boot项目打包成war包,并成功地部署到服务器上。 ... [详细]
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • Java在运行已编译完成的类时,是通过java虚拟机来装载和执行的,java虚拟机通过操作系统命令JAVA_HOMEbinjava–option来启 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • 本文详细介绍了在ASP.NET中获取插入记录的ID的几种方法,包括使用SCOPE_IDENTITY()和IDENT_CURRENT()函数,以及通过ExecuteReader方法执行SQL语句获取ID的步骤。同时,还提供了使用这些方法的示例代码和注意事项。对于需要获取表中最后一个插入操作所产生的ID或马上使用刚插入的新记录ID的开发者来说,本文提供了一些有用的技巧和建议。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 如何实现JDK版本的切换功能,解决开发环境冲突问题
    本文介绍了在开发过程中遇到JDK版本冲突的情况,以及如何通过修改环境变量实现JDK版本的切换功能,解决开发环境冲突的问题。通过合理的切换环境,可以更好地进行项目开发。同时,提醒读者注意不仅限于1.7和1.8版本的转换,还要适应不同项目和个人开发习惯的需求。 ... [详细]
author-avatar
梦幻00草根_831
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有