`

java反射与内省(Introspector)

    博客分类:
  • Java
 
阅读更多

一、java反射机制

    JAVA反射机制是在运行状态中,对于任意一个类,都能够得到这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

    用一句比较白的话来概括,反射就是让你可以通过名称来得到对象 ( 类,属性,方法 ) 的技术。例如我们可以通过类名来生成一个类的实例;知道了方法名,就可以调用这个方法;知道了属性名就可以访问这个属性的值。

Java反射机制主要提供了以下功能:

1、为一个类生成对应的Class对象

   运用(已知对象) getClass():Object类中的方法,每个类都拥有此方法。

        如:String str=new String();Class strClass=str.getClass();

     运用(已知子类的class) Class.getSuperclass():Class类中的方法,返回该Class的父类的Class;

     运用(已知类全名)Class.forName()静态方法

     运用(已知类)类名.class

2、通过类名来构造一个类的实例

   a、调用无参的构造函数:

   Class newoneClass = Class.forName(类全名);

   newoneClass.newInstance();

   b、调用有参的构造函数:我们可以自定义一个函数。 

   public Object newInstance(String className, Object[] args) throws Exception {

        //args为参数数组

        Class newoneClass = Class.forName(className); 

        //得到参数的Class数组(每个参数的class组成的数组),由此来决定调用那个构造函数

        Class[] argsClass = new Class[args.length];

        for (int i = 0, j = args.length; i < j; i++) {

            argsClass[i] = args[i].getClass();

        } 

        Constructor cons = newoneClass.getConstructor(argsClass); //根据argsClass选择函数

        return cons.newInstance(args); //根据具体参数实例化对象。

   }

3、得到某个对象的属性

   a、非静态属性:首先得到class,然后得到这个class具有的field,然后以具体实例为参数

      调用这个field

   public Object getProperty(Object owner, String fieldName) throws Exception {

       Class ownerClass = owner.getClass();//首先得到class

       Field field = ownerClass.getField(fieldName);

       //然后得到这个class具有的field,也可以通过getFields()得到所有的field

       Object property = field.get(owner);

       //owner指出了取得那个实例的这个属性值,

         如果这个属性是非公有的,这里会报IllegalAccessException。

       return property;

   }

   b、静态属性:

      只有最后一步不同,由于静态属性属于这个类,所以只要在这个类上调用这个field即可

      Object property = field.get(ownerClass);

4、执行某对象的方法

  public Object invokeMethod(Object owner, String methodName, 

                             Object[] args) throws Exception { 

     Class ownerClass = owner.getClass(); //也是从class开始的

     //得到参数的class数组,相当于得到参数列表的类型数组,来取决我们选择哪个函数。

     Class[] argsClass = new Class[args.length]; 

     for (int i = 0, j = args.length; i < j; i++) {

         argsClass[i] = args[i].getClass();

     } 

     //根据函数名和函数类型来选择函数

     Method method = ownerClass.getMethod(methodName, argsClass); 

     return method.invoke(owner, args);//具体实例下,具体参数值下调用此函数

  }

5、执行类的静态方法

   和上面的相似只是最后一行不需要指定具体实例

   return method.invoke(null, args);

6、判断是否为某个类的实例

   public boolean isInstance(Object obj, Class cls) {

        return cls.isInstance(obj);

   }

二、java内省机制

    内省是 Java 语言对 Bean 类属性、事件的一种处理方法(也就是说给定一个javabean对象,我们就可以得到/调用它的所有的get/set方法)。例如类 A 中有属性 name, 那我们可以通过 getName,setName 来得到其值或者设置新的值。通过 getName/setName 来访问 name 属性,这就是默认的规则。 Java 中提供了一套 API 用来访问某个属性的 getter/setter 方法,通过这些 API 可以使你不需要了解这个规则,这些 API 存放于包 java.beans 中。

    一般的做法是通过类 Introspector 来获取某个对象的 BeanInfo 信息,然后通过 BeanInfo 来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后我们就可以通过反射机制来调用这些方法。下面我们来看一个例子,这个例子把某个对象的所有属性名称和值都打印出来:

//定义一个javabean

public class PersonBean {

   public String name;

   public String[] childname;

   public String[] getChildname() {

      return childname;

   }

   public void setChildname(String[] childname) {

      this.childname = childname;

   }

   public String getName() {

      return name;

   }

   public void setName(String name) {

      this.name = name;

   }

}

 

//定义一个测试类,来进行一下set和get方法的调用举例

import java.beans.BeanInfo;

import java.beans.IntrospectionException;

import java.beans.Introspector;

import java.beans.PropertyDescriptor;

import java.lang.reflect.InvocationTargetException;

public class TestIntrospector {

   public static void main(String[] args) throws IllegalArgumentException,

                           IllegalAccessException, SecurityException, NoSuchMethodException,

                           InvocationTargetException, IntrospectionException {

  //演示一下get方法的调用

      //初始化一个javabean对象

      PersonBean pb=new PersonBean();

      pb.setName("kangjian");

      String[] childs=new String[]{"kk","jj","nn"};

      pb.setChildname(childs);

      //将该javabean中的属性放入到BeanInfo中。第二个参数为截止参数,表示截止到此类之前,

       不包括此类。如果不设置的话,那么将会得到本类以及其所有父类的info。

      BeanInfo bi=Introspector.getBeanInfo(pb.getClass(), Object.class);

      //将每个属性的信息封装到一个PropertyDescriptor形成一个数组

        其中包括属性名字,读写方法,属性的类型等等

      PropertyDescriptor[] pd=bi.getPropertyDescriptors();

      //演示如何get

      for (int i = 0; i < pd.length; i++) {

         if(pd[i].getPropertyType().isArray())  //getPropertyType得到属性类型。

         {

            //getReadMethod()得到此属性的get方法----Method对象,然后用invoke调用这个方法

            String[] result=(String[]) pd[i].getReadMethod().invoke(pb, null);

            System.out.println(pd[i].getName()+":");//getName得到属性名字

            for (int j = 0; j < result.length; j++) {

               System.out.println(result[j]);

            }

         }

         else

         {

            System.out.println(pd[i].getName()+":"+pd[i].getReadMethod().invoke(pb, null));

         }

      }

  //演示一下set方法的调用

      //初始化一个尚未set的javabean

      PersonBean pb0=new PersonBean();

      //模拟一个数据(数据名字和javabean的属性名一致)

      String name="luonan";

      String[] childname=new String[]{"luo","nan"};

 

      BeanInfo bi0=Introspector.getBeanInfo(pb0.getClass(), Object.class);

      PropertyDescriptor[] pd0=bi0.getPropertyDescriptors();

 

      for (int i = 0; i < pd0.length; i++) {

         if(pd0[i].getPropertyType().isArray())

         {

            if(pd0[i].getName().equals("childname"));

            {

               if(pd0[i].getPropertyType().getComponentType().equals(String.class))

               {//getComponentType()可以得到数组类型的元素类型

                  //getWriteMethod()得到此属性的set方法---Method对象,然后用invoke调用这个方法

                  pd0[i].getWriteMethod().invoke(pb0,new Object[]{childname});

               }

            }

         }

         else

         {

            if(pd0[i].getName().equals("name"));

            {

               pd0[i].getWriteMethod().invoke(pb0,name);

            }

         }

      }

 

      System.out.println(pb0.getName());

      String[] array=pb0.getChildname();

      for (int i = 0; i < array.length; i++) {

         System.out.println(array[i]);

      }

   }

更难的应用请参考http://hi.baidu.com/shunxinyangkun/blog/item/dd2c4ff46ae61aeb7609d7ba.html

三、关于内省的思考

    struts2的action(还有struts1的formbean)就是这么实现的。

    前台的form标签具有一些属性(在配置文件中知道这个form提交到那个action,而这个action有和这个form相对应的属性及其 get/set),提交以后,由struts的servlet拦下来转发给某个具体的action.而在转发给action之前struts通过内省的方式将form中的值set到了action中去。

    其实只要有个set**或者get**,内省就会理解为存在这样的**属性,这样可以方便我们把 Bean 类通过一个接口来定义而不用去关心具体实现,不用去关心 Bean 中数据的存储。比如我们可以把所有的 getter/setter 方法放到接口里定义,但是真正数据的存取则是在具体类中去实现,这样可提高系统的扩展性。

四、总结

    将 Java 的反射以及内省应用到程序设计中去可以大大的提供程序的智能化和可扩展性。有很多项目都是采取这两种技术来实现其核心功能,例如我们前面提到的 Struts ,还有用于处理 XML 文件的 Digester 项目,其实应该说几乎所有的项目都或多或少的采用这两种技术。在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性。

分享到:
评论

相关推荐

    JAVA的内省机制(introspector)与反射机制(reflection).docx

    "JAVA的内省机制(introspector)与反射机制(reflection)" JAVA 的内省机制(introspector)和反射机制(reflection)是两个重要的概念,在 JAVA 编程中扮演着至关重要的角色。那么,什么是内省机制和反射机制?它们...

    Java 内省introspector相关原理代码解析

    `Introspector` 是Java内省的核心类,它提供了一系列的方法来获取关于JavaBean的信息。例如,`Introspector.getBeanInfo(Class)` 方法可以获取指定类的BeanInfo,这包含了关于类的属性(PropertyDescriptor)、事件...

    Java中的内省与反射.doc

    ### Java中的内省与反射详解 #### 一、引言 在深入了解Java语言的过程中,我们会遇到两个重要的概念:反射和内省。这两种技术在实际开发中有着广泛的应用,尤其是在需要进行动态操作或元数据处理的场景下尤为重要...

    Java 内省(Introspector)深入理解

    Java内省(Introspector)是Java语言提供的一种机制,用于在运行时分析Java对象的属性、方法和事件。这个特性对于开发人员来说非常有用,因为它允许程序动态地检查和修改对象的状态,而不必了解对象的具体实现细节。...

    用反射和内省技术实现简单SpringIOC

    内省(Introspection)则是Java提供的一种用于获取和修改对象属性的能力,主要通过java.beans包中的Introspector和PropertyDescriptor类实现。它主要用于处理JavaBeans规范的对象,可以获取Bean的属性、事件和方法...

    java-beans-lite:轻量级且快速的java.beans.Introspector实现

    轻巧,快速的java.beans.Introspector重新实现,用于消除对Bean Introspection的java.desktop模块的依赖。 问题 JDK 9中引入的模块封装了Java标准库中的所有AWT,Swing,Image和Sound软件包。 除此之外,它还包含带...

    Java反射中java.beans包学习总结.docx

    在Java编程语言中,反射(Reflection)是一种强大的工具,它允许程序在运行时检查和操作类、接口、字段和方法的信息。JavaBeans包(java.beans)提供了与JavaBeans组件交互的相关类和接口,这些组件是Java面向对象...

    使用betwixt进行xml与java类之间的互转解析--代码

    本篇文章将深入探讨如何使用Betwixt库实现XML与Java类之间的互转解析。 首先,我们需要理解Betwixt的基本概念。Betwixt是一个XML绑定工具,它允许开发者通过注解或XML配置文件来映射XML结构与Java类的属性。这种...

    java深度复制源代码

    - **反射机制**:该工具类大量使用了Java反射API(如`Introspector.getBeanInfo()`、`PropertyDescriptor`等)来获取和设置JavaBean的属性。 - **泛型**:使用了泛型`&lt;T&gt;`,使得该工具类可以应用于任意类型的...

    Java基础学习43.pdf

    内省机制(Introspection)是Java提供的一种工具,用于获取和操作JavaBeans对象的属性。例如,通过Introspector、BeanInfo和PropertyDescriptor等类,可以方便地获取和设置JavaBean的属性值。 BeanUtils是Apache ...

    用于提高java技术的资料 javaBeAN

    2. **Introspector**:Java提供了一个名为`Introspector`的工具类,它能自动查找JavaBean的属性、方法和事件,便于操作JavaBean。 3. **JSP和Servlet中的JavaBean**:在Web开发中,JavaBean常被用作JSP页面和...

    JavaBean及内省文档

    通过使用 `java.beans.Introspector` 类,可以在运行时发现 JavaBean 的属性并进行操作。 在 J2EE 应用中,内省机制常用于框架自动识别 JavaBean 的属性,并根据这些属性进行相应的操作,例如:在容器中自动注入...

    openbeans1.0.zip

    //import java.beans.Introspector; //import java.beans.PropertyDescriptor; import com.googlecode.openbeans.BeanInfo; import com.googlecode.openbeans.IntrospectionException; import ...

    java.beans.*

    6. **Introspection**: `java.beans.Introspector` 类提供了一种动态分析 Java 类的方法,以便发现其属性、事件和动作。在 Android 中,虽然没有直接使用 Bean 的 introspection,但反射(Reflection)在某些情况下...

    Thinking in Java 中文第四版+习题答案

    15.6 Java与CGI的沟通 15.6.1 CGI数据的编码 15.6.2 程序片 15.6.3 用C++写的CGI程序 15.6.4 POST的概念 15.7 用JDBC连接数据库 15.7.1 获得学习示例 15.7.2 查找程序的GUI版本 15.7.3 JDBC API为何如何复杂 15.8 ...

    xml转换java对象

    1. **配置映射**: 使用Betwixt,你需要创建一个`BeanIntrospector`实例,并通过`introduceBean()`方法指定XML与Java对象的映射规则。通常,Betwixt会尝试基于Java类的公共字段和方法自动建立映射,但也可以通过注解...

    如何利用反射批量修改java类某一属性的代码详解

    反射机制在 Java 中的应用 - 批量修改 Java 类某一属性 在 Java 中,反射机制是一种非常重要的技术,它可以在运行时动态地获取和修改类的信息。在本文中,我们将详细介绍如何利用反射机制批量修改 Java 类某一属性...

    使用struts2的Introspector做日志

    Struts2是一个流行的Java web框架,它为构建MVC(模型-...总之,通过Struts2的Introspector和拦截器机制,开发者可以轻松地集成日志功能,提供透明且全面的日志记录,这对于优化和维护Java web应用来说是非常有价值的。

    java白皮书 java电子书

    9. **JavaBeans规范**:解释JavaBean的属性、构造函数、事件和设计模式,以及如何通过Introspector和BeanInfo类来访问和修改Bean的属性。 10. **Java Swing和JavaFX**:这两是Java的图形用户界面(GUI)工具包,...

    java 编程入门思考

    15.6 Java与CGI的沟通 15.6.1 CGI数据的编码 15.6.2 程序片 15.6.3 用C++写的CGI程序 15.6.4 POST的概念 15.7 用JDBC连接数据库 15.7.1 获得学习示例 15.7.2 查找程序的GUI版本 15.7.3 JDBC API为何如何复杂 15.8 ...

Global site tag (gtag.js) - Google Analytics