最新文章专题视频专题问答1问答10问答100问答1000问答2000关键字专题1关键字专题50关键字专题500关键字专题1500TAG最新视频文章推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37视频文章20视频文章30视频文章40视频文章50视频文章60 视频文章70视频文章80视频文章90视频文章100视频文章120视频文章140 视频2关键字专题关键字专题tag2tag3文章专题文章专题2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章专题3
当前位置: 首页 - 正文

java反射机制详解与应用

来源:动视网 责编:小OO 时间:2025-09-22 22:54:16
文档

java反射机制详解与应用

java有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种“看透class”的能力(theabilityoftheprogramtoexamineitself)被称为introspecti
推荐度:
导读java有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种“看透class”的能力(theabilityoftheprogramtoexamineitself)被称为introspecti
java有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种“看透class”的能力(the ability of the program to examine itself)被称为introspection(内省、内观、反省)。Reflection和introspection是常被并提的两个术语。

       这个机制允许程序在运行时透过Reflection APIs取得任何一个已知名称的class的内部信息,包括其modifiers(诸如public, static 等等)、superclass(例如Object)、实现之interfaces(例如Cloneable),也包括fields和methods的所有信息,并可于运行时改变fields内容或唤起methods。

       目前好多框架都会用到java的反射机制。比如struts2,sping,hibernate。

如果我们不用struts2,自己写一个类似的功能也是可以实现的,比如浏览器通过HTTP发送数据,而这些数据都是字符串,我们接受到这些字符串时, 可以通过反射去构造一个对象(通过做成框架的功能),这样就可以用对象的get和set方法了,而不用原始的getPeremter方法。事实上, 在struts2出来之前,我们又不想用struts1的ActionForm就做过这样项目。

一、Class object 的产生方式有以下几种。

1、运用getClass()

注:每个class 都有此函数

    

String str = "abc";

Class c1 = str.getClass();

2、运用static method Class.forName()(最常被使用)

Class c1 = Class.forName ("java.lang.String");

Class c2 = Class.forName ("java.awt.Button");

3、运用.class 语法

    

Class c1 = String.class;

Class c2 = java.awt.Button.class;

4、运用primitive wrapper classes的TYPE 语法

Class c1 = Integer.TYPE;

Class c2 = Long.TYPE;

二、Java类反射中的主要方法

对于以下三类组件中的任何一类来说 -- 构造函数、字段和方法 -- java.lang.Class 提供四种的反射调用,以不同的方式来获得信息。调用都遵循一种标准格式。以下是用于查找构造函数的一组反射调用:

Constructor getConstructor(Class[] params) -- 获得使用特殊的参数类型的公共构造函数,

Constructor[] getConstructors() -- 获得类的所有公共构造函数

Constructor getDeclaredConstructor(Class[] params) -- 获得使用特定参数类型的构造函数(与接入级别无关)

Constructor[] getDeclaredConstructors() -- 获得类的所有构造函数(与接入级别无关) 

获得字段信息的Class 反射调用不同于那些用于接入构造函数的调用,在参数类型数组中使用了字段名:

Field getField(String name) -- 获得命名的公共字段

Field[] getFields() -- 获得类的所有公共字段

Field getDeclaredField(String name) -- 获得类声明的命名的字段

Field[] getDeclaredFields() -- 获得类声明的所有字段 

用于获得方法信息函数:

Method getMethod(String name, Class[] params) -- 使用特定的参数类型,获得命名的公共方法

Method[] getMethods() -- 获得类的所有公共方法

Method getDeclaredMethod(String name, Class[] params) -- 使用特写的参数类型,获得类声明的命名的方法

Method[] getDeclaredMethods() -- 获得类声明的所有方法 

三、以下是代码实例:

import java.lang.reflect.Constructor;

import java.lang.reflect.Field;

import java.lang.reflect.Method;

public class ReflectTester {

    /**

    * 通过java的反射机制获取类的所有属性和方法

    */

    public void test1() {

        try {

            Class c = Class.forName("demo1.client.Customer");

            System.out.println("属性:");

            Field f[] = c.getDeclaredFields();

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

                System.out.println(f[i].getName());

            }

            System.out.println("方法:");

            Method m[] = c.getDeclaredMethods();

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

                System.out.println(m[i].toString());

            }

        } catch (Throwable e) {

            System.err.println(e);

        }

    }

    /**

    * 通过java的反射机制动态修改对象的属性

    * @param o

    */

    public void test2(Customer o) {

        try {

            Class c = o.getClass();

            //getMethod方法第一个参数指定一个需要调用的方法名称,第二个参数是需要调用方法的参数类型列表,如无参数可以指定null,该方法返回一个方法对象 

            Method sAge = c.getMethod("setAge", new Class[] { int.class });

            Method gAge = c.getMethod("getAge", null);

            Method sName = c.getMethod("setName", new Class[] { String.class });

            //动态修改Customer对象的age

            Object[] args1 = { new Integer(25) };

            sAge.invoke(o, args1);

            //动态取得Customer对象的age

            Integer AGE = (Integer) gAge.invoke(o, null);

            System.out.println("the Customer age is: " + AGE.intValue());

            //动态修改Customer对象的name

            Object[] args2 = { new String("李四") };

            sName.invoke(o, args2);

        } catch (Throwable e) {

            System.err.println(e);

        }

    }

    /**

    * 通过java的反射机制做一个简单对象的克隆

    * @param o

    * @return

    */

    public Object test3(Customer o) {

        Object o2 = null;

        try {

            Class c = o.getClass();

            //通过默认构造方法创建一个新的对象 

            o2 = c.getConstructor(new Class[] {}).newInstance(

                    new Object[] {});

            Field fields[] = c.getDeclaredFields();

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

                Field field = fields[i];

                String fieldName = field.getName();

                String firstLetter = fieldName.substring(0, 1).toUpperCase();

                // 获得和属性对应的getXXX()方法的名字 

                String getMethodName = "get" + firstLetter + fieldName.substring(1);

                // 获得和属性对应的setXXX()方法的名字 

                String setMethodName = "set" + firstLetter + fieldName.substring(1); 

                // 获得和属性对应的getXXX()方法 

                Method getMethod = c.getMethod(getMethodName, new Class[] {}); 

                // 获得和属性对应的setXXX()方法 

                Method setMethod = c.getMethod(setMethodName, new Class[] { field.getType() }); 

                // 调用原对象的getXXX()方法 

                Object value = getMethod.invoke(o, new Object[] {});

               // 调用拷贝对象的setXXX()方法 

                setMethod.invoke(o2, new Object[] { value });

            }

        } catch (Throwable e) {

            System.err.println(e);

        }

        return o2;

    }

    public static void main(String[] args) throws Exception {

        ReflectTester t = new ReflectTester();

        t.test1();

        Customer customer = new Customer();

        customer.setAge(20);

        customer.setName("张三");

        System.out.println("调用前name: " + customer.getName());

        System.out.println("调用前age: " + customer.getAge());

        t.test2(customer);

        System.out.println("调用后name: " + customer.getName());

        System.out.println("调用后age: " + customer.getAge());

        Customer customer2 = (Customer)t.test3(customer);

        System.out.println("克隆对象的name: " + customer2.getName());

        System.out.println("克隆对象的age: " + customer2.getAge());

    }

}

class Customer {

    private long id;

    private String name;

    private int age;

    public Customer() {

    }

    public Customer(String name, int age) {

        this.name = name;

        this.age = age;

    }

    public long getId() {

        return id;

    }

    public void setId(long id) {

        this.id = id;

    }

    public String getName() {

        return name;

    }

    public void setName(String name) {

        this.name = name;

    }

    public int getAge() {

        return age;

    }

    public void setAge(int age) {

        this.age = age;

    }

运行结果:

属性:

id

name

age

方法:

public java.lang.String demo1.client.Customer.getName()

public long demo1.client.Customer.getId()

public void demo1.client.Customer.setName(java.lang.String)

public void demo1.client.Customer.setAge(int)

public int demo1.client.Customer.getAge()

public void demo1.client.Customer.setId(long)

调用前name: 张三

调用前age: 20

the Customer age is: 25

调用后name: 李四

调用后age: 25

克隆对象的name: 李四

克隆对象的age: 25

文档

java反射机制详解与应用

java有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种“看透class”的能力(theabilityoftheprogramtoexamineitself)被称为introspecti
推荐度:
  • 热门焦点

最新推荐

猜你喜欢

热门推荐

专题
Top