Java反射机制实例

java有着一个非常突出的动态相关机制:Reflection。这个字的意思是反射、映象、倒影,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种看透class”的能力(the ability of the program to examine itself)被称为introspection(内省、内观、反省)。Reflectionintrospection是常被并提的两个术语。
      
这个机制允许程序在运行时透过Reflection APIs取得任何一个已知名称的class的内部信息,包括其modifiers(诸如public, static 等等)、superclass(例如Object)、实现之interfaces(例如Cloneable),也包括fieldsmethods的所有信息,并可于运行时改变fields内容或唤起methods

      
目前好多框架都会用到java的反射机制。比如struts2,sping,hibernate
如果我们不用struts2,自己写一个类似的功能也是可以实现的,比如浏览器通过HTTP发送数据,而这些数据都是字符串,我们接受到这些字符串时, 可以通过反射去构造一个对象(通过拦截器做成框架的功能),这样就可以用对象的getset方法了,而不用原始的getPeremter方法。事实上, struts2出来之前,我们又不想用struts1ActionForm就做过这样项目。
一、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 classesTYPE 语法

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,反射机制,影像)