java反射机制详解以及应用

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

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值