反射是 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. 各种食物的类图

例如 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> T getBean(){
T ret=null;
//read informations from foods.xml by dom4j
SAXReader 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 class
Hashtable<String,String> hashtbl=new Hashtable<String,String>();
//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 value
List list1=element.selectNodes("property");
Iterator<Element> 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 hashtable
hashtbl.put(propertyName, propertyType+";"+propertyValue);
}
//call the method to create a object
ret=newInstance(name,className,hashtbl);
return ret;
}
private <T> T newInstance(String name,String className,
Hashtable<String,String> hashtbl){
T ret=null;
try {
//create object by reflection
ret=(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 reflection
type=Class.forName(hashtbl.get(key).split(";")[0]);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
try {
//get method object by reflection
Method m=ret.getClass().getMethod(methodName, type);
//call method by reflection
m.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 文件的配置
<?xml version="1.0" encoding="UTF-8"?>
<foods>
<choice>vegetables</choice>
<vegetables>
<vegetable name="tomato" class="vegetables.Tomato">
<property name="color" type="java.lang.String">
<value>red</value>
</property>
</vegetable>
<!--
<vegetable name="potato" class="vegetables.Potato">
<property name="shape" type="java.lang.String">
<value>nice</value>
</property>
<property name="size" type="java.lang.String">
<value>big</value>
</property>
</vegetable>-->
</vegetables>
<fruits>
<fruit name="pear" class="fruits.Pear">
<property name="shape" type="java.lang.String">
<value>nice</value>
</property>>
</fruit>
</fruits>
</foods>

如果我们决定买西红柿,因为它属于蔬菜,所以设定 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 和反射的这些功能会帮助我们更快,更方便地实现一些 动态代码。

参考资料

结合反射与 XML 实现 Java 编程的动态性的更多相关文章

  1. Java编程 的动态性,第 2部分: 引入反射--转载

    在“ Java编程的动态性,第1部分,”我为您介绍了Java编程类和类装入.该篇文章介绍了一些Java二进制类格式的相关信息.这个月我将阐述使用Java反射API来在运行时接入和使用一些相同信息的基础 ...

  2. Java 编程的动态性,第 8 部分: 用代码生成取代反射--转载

    既然您已经看到了如何使用 Javassist 和 BCEL 框架来进行 classworking (请参阅 本系列以前的一组文章), 我将展示一个实际的 classworking 应用程序.这个应用程 ...

  3. Java 编程的动态性,第 7 部分: 用 BCEL 设计字节码--转载

    在本系列的最后三篇文章中,我展示了如何用 Javassist 框架操作类.这次我将用一种很不同的方法操纵字节码——使用 Apache Byte Code Engineering Library (BC ...

  4. Java 编程的动态性 第1 部分: 类和类装入--转载

    原文地址:http://www.ibm.com/developerworks/cn/java/j-dyn0429/ 本文是这个新系列文章的第一篇,该系列文章将讨论我称之为 Java 编程的动态性的一系 ...

  5. Java 编程的动态性,第3部分: 应用反射--转载

    在 上个月的文章中,我介绍了Java Reflection API,并简要地讲述了它的一些基本功能.我还仔细研究了反射的性能,并且在文章的最后给出了一些指导方针,告诉读者在一个应用程序中何时应该使用反 ...

  6. Java 编程的动态性,第 6 部分: 利用 Javassist 进行面向方面的更改--转载

    本系列的 第 4 部分和 第 5 部分讨论了如何用 Javassist 对二进制类进行局部更改.这次您将学习以一种更强大的方式使用该框架,从而充分利用 Javassist 对在字节码中查找所有特定方法 ...

  7. Java 编程的动态性,第 5 部分: 动态转换类--转载

    在第 4 部分“ 用 Javassist 进行类转换”中,您学习了如何使用 Javassist 框架来转换编译器生成的 Java 类文件,同时写回修改过的类文件.这种类文件转换步骤对于做出持久变更是很 ...

  8. Java 编程的动态性, 第4部分: 用 Javassist 进行类转换--转载

    讲过了 Java 类格式和利用反射进行的运行时访问后,本系列到了进入更高级主题的时候了.本月我将开始本系列的第二部分,在这里 Java 类信息只不过是由应用程序操纵的另一种形式的数据结构而已.我将这个 ...

  9. 【Java编程】写入、读取、遍历Properties文件

    在Java开发中通常我们会存储配置參数信息到属性文件.这种属性文件能够是拥有键值对的属性文件,也能够是XML文件.关于XML文件的操作,请參考博文[Java编程]DOM XML Parser 解析.遍 ...

  10. Java编程之反射中的注解详解

    "注解"这个词,可谓是在Java编程中出镜率比较高,而且也是一个老生常谈的话题.我们之前在聊Spring相关的东西时,注解是无处不在,之前我们简单的聊过一些"注解&quo ...

随机推荐

  1. Kubernetes-3.2:kubespray安装高可用k8sv1.20.2集群及常见报错解决

    kubespray安装高可用k8s集群 环境介绍 系统环境 主机名 / IP地址 角色 内核版本 CentOS 7.6.1810 master1 / 192.168.181.252 master &a ...

  2. Flask 从开发到部署

    整理一下怎么开发flask程序应部署到生产环境中 1. 第一个flask 程序 myapp.py from flask import Flask app = Flask(__name__) @app. ...

  3. 有哪些让你「 爽到爆炸 」的 Windows 软件?

    前言 本文源于知乎的一个提问,如标题所示:有哪些让你「 爽到爆炸 」的 Windows 软件?今天大姚给大家分享6款C#/.NET开源且免费的Windows软件,希望可以帮助大家提高学习.开发.办公效 ...

  4. 物体检测序列之一:ap, map

    准确率(Precision),也叫正确预测率(positive predictive value),在模式识别.信息检索.机器学习等研究应用领域,准确率用来衡量模型预测的结果中相关或者正确的比例.而召 ...

  5. sql 查询表数据

    SELECT s.Name AS SchemaName, t.Name AS TableName, p.rows AS RowCounts--, --CAST(ROUND((SUM(a.used_pa ...

  6. MySQL服务无法启动 服务没有报告任何错误

    安装MYSQL后 启动服务 出现错误 在启动MySQL服务时 出现该报错 解决方法: 将原本在MySQL根目录下的my.ini文件移动到bin目录下(my.ini文件参考:这里)    删除根目录下的 ...

  7. [快速阅读八] HDR->LDR:Matlab中tonemapfarbman函数的解析和自我实现。

    最近受朋友的委托,想自己实现Matlab里的一个HDR转LDR的函数,函数名是tonemapfarbman,乘着十一假期,稍微浏览下这个函数,并做了一点C++的实现和优化. 为了看到这个函数的效果,需 ...

  8. USB协议详解第4讲(USB描述符-标准配置描述符)

    1.USB描述符 USB描述符有设备描述符.标准配置描述符.接口描述符.端点描述符.字符串描述符,HID设备有HID描述符.报告描述符和物理描述符.今天主要是学习USB标准配置描述符的组成. 2.标准 ...

  9. android 性能优化 -systrace

    简介: Systrace允许监视和跟踪Android系统的行为(trace).它会指明系统都在哪些工作上花费时间.CPU周期都用在哪里,甚至可以看到每个线程.进程在指定时间内都在干嘛.它同时还会突出观 ...

  10. .NET周刊【10月第1期 2024-10-06】

    国内文章 基于DPAPI+RDP技术实现本地打开远程程序,并映射到本地机器桌面上 https://www.cnblogs.com/weskynet/p/18445584 该教程讲述如何使用Remote ...