`

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

阅读更多
转载于张 立达
2009 年 12 月 30 日
反射是 Java 语言被视为动态或准动态语言的一个关键性质,结合反射和 XML 会帮助我们更快、更方便地实现一些动态代码,从而解决编程中可能遇到的不确定问题。本文将结合反射与 XML 对 Java 编程的动态性进行深入浅出的讨论。在理解本文的思想之后,您可以将其应用到程序中以创建灵活的代码。
引言
在现实生活中,经常会发生这种情况。我要去商场买菜来准备晚餐,我不知道买什么,但是进入商场之前我会随手拿一个购物筐来装最终决定要买的物品,这样不管我买什么都可以放入其中,结账之后就可以带回家准备晚饭。在开发程序过程中,也会遇到类似情况。有时我们不能确定类的名称、类有哪些属性以及属性的值等,这些内容只有到运行的时候才能确定。这种情况下,不能将类的名称直接固化,那么怎样才能解决这种问题呢?其实开发程序与现实生活有相似之处。在编程之中,购物筐就好比事先准备好的通用接口,这个接口可以用反射的机制来实现,而购物筐中的菜可以由 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 和反射的这些功能会帮助我们更快,更方便地实现一些 动态代码。
分享到:
评论
1 楼 anttribe 2010-10-02  

相关推荐

    java 反射对XML的解析

    总的来说,Java反射机制与XML的结合,使得我们在编写代码时可以具有更高的灵活性和动态性,特别是在处理配置文件、插件系统或者需要在运行时动态加载和操作类的情况下。但同时,反射也带来了性能开销和安全风险,...

    解析并创建XML文件和Java的反射

    Java的反射机制则是Java编程语言提供的一种强大的功能,允许程序在运行时检查类、接口、字段和方法的信息,并能够动态调用方法和创建对象。 在解析XML文件时,Java提供了多种API,其中最常用的是DOM(Document ...

    accp xml java反射 PPT

    【Java反射】Java反射是Java编程语言的一项高级特性,它允许程序在运行时动态地获取类的信息并操作类的对象。通过反射,开发者可以实现元编程,即在运行时检查类、接口、字段和方法,创建和调用对象。在PPT中,可能...

    反射在Java Swing编程中的应用

    在Swing编程中,反射的应用可以帮助开发者实现更加灵活和动态的UI设计。 反射的应用主要体现在以下几个方面: 1. **动态创建和操作组件**:Swing组件通常需要在代码中预先定义和实例化。然而,通过反射,我们可以...

    xml反射基本用法........

    XML反射是Java编程中一种强大...总结起来,XML反射结合了XML的数据表示能力和Java反射的动态性,使我们能够在运行时根据XML内容操作Java对象。理解和熟练掌握这一技术,能够极大地提升我们的编程效率和代码的可扩展性。

    XML解析与反射应用联合.zip

    "XML解析与反射应用联合"的主题,意味着这个压缩包可能包含了一个示例项目,展示了如何结合XML解析和反射来实现灵活的配置和动态行为。例如,可能有一个XML配置文件定义了某些类的实例化和它们应执行的方法,然后...

    java 根据javaBean反射自定义导出 excel、xml、pdf、csv

    在Java编程中,根据JavaBean反射来实现数据的导出,如Excel、XML、PDF和CSV格式,是一种常见的数据处理技术。JavaBean是Java中的一种设计模式,它提供了一种标准的方式来封装对象属性,便于数据操作。反射机制则是...

    Java编程 100例

    理解类的构造器、方法、访问修饰符以及如何通过继承和接口实现多态性,是进阶Java编程的关键。 3. **异常处理**:Java中的try-catch-finally结构用于捕获和处理运行时错误,是编写健壮代码的重要手段。学习如何正确...

    java编程深入研究

    《Java编程深入研究》涵盖了Java开发的多个重要方面,旨在帮助开发者从基础到高级,全面理解和掌握这门广泛使用的编程语言。在这个过程中,我们将探讨Java的环境配置、服务器的运用,以及一系列实用示例。 首先,...

    java 解析xml类映射数据库 数据库操作 反射机制 源码

    在Java编程中,XML(eXtensible Markup Language)是一种常用的数据交换格式,常用于存储配置信息或数据映射。本程序集的核心是利用Java解析XML文件,将XML中的数据映射到数据库,并通过反射机制进行数据库操作。...

    java反射实现Object转json

    Java反射是Java编程语言中的一个强大工具,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射主要用于在运行时分析类和对象的信息,包括类名、方法名、参数类型等。对于"java...

    利用Java的反射与代理实现IOC模式

    Java的反射机制是Java编程语言的核心特性之一,它允许程序在运行时检查和操作类、接口、对象等的内部结构。反射机制使得开发者能够在运行时动态地创建对象、调用方法、访问字段,以及发现类的元数据。通过反射,我们...

    Java基础精品课22-xml与反射.zip

    本课程“Java基础精品课22-xml与反射”聚焦于两个关键概念:XML(可扩展标记语言)和Java反射。XML是用于存储和传输数据的标准格式,而反射则是Java提供的一种强大的动态类型功能。 XML,全称为eXtensible Markup ...

    java反射机制 字符串——java对象

    总的来说,Java反射机制和字符串操作是Java开发中不可或缺的部分,它们允许我们实现动态编程,增强程序的灵活性和可扩展性。通过结合XML序列化和解析,我们可以方便地在Java对象和XML数据之间进行转换,这对于数据...

    基于Java的XML解析与反射设计模式.doc

    Java作为面向对象的编程语言,因其动态性、安全性以及丰富的库支持,成为了实现Web服务和XML解析的首选工具。 XML作为一种通用的数据交换格式,其解析方式主要有DOM、SAX和基于库的方法,如XStream。DOM解析器将...

    基于Java的XML解析与反射设计模式.pdf

    "基于Java的XML解析与反射设计模式" 在软件开发中,XML解析与反射设计模式是两个非常重要的概念。随着计算机时代的蓬勃发展,Internet的普及给社会以及人民的生活带来了巨大的影响。同时,B/S结构的多系统相互访问...

    通过Java annotation以及反射机制实现不同类型通用的数据库访问接口

    在Java编程中,注解(Annotation)和反射(Reflection)是两个非常强大的工具,它们能够帮助我们实现更灵活、更具动态性的代码。本篇将详细探讨如何利用这两者来实现一个通用的数据库访问接口,以适应不同类型的...

    采用反射技术和XML的通用实体类

    在Java编程中,通用实体类是一种设计模式,用于创建可复用的对象模型,这些对象能够代表数据库中的表或XML文档中的元素。这种设计允许开发者在不修改代码的情况下处理不同的数据结构,提高了代码的灵活性和可维护性...

    java反射实例

    Java反射是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在标题“Java反射实例”中,我们聚焦于如何在实际编程中应用这一特性。通过反射,我们可以动态地...

    Java编程思想 中文文档

    《Java编程思想》是 Bruce Eckel 的经典著作,这本书深入浅出地介绍了Java语言的核心概念和技术,对于初学者和有经验的程序员来说都是极好的学习资源。中文文档的出现使得更多的中国开发者能够无障碍地理解这本权威...

Global site tag (gtag.js) - Google Analytics