• 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
  • 相关阅读:
    lxml库
    requests库基本使用
    Xpath Helper的使用
    Class.forName()的作用(转)
    JDBC 连接数据库
    IDEA 的 Othere Settings(Default settings)消失了?(转)
    servletContext.getRealPath(String)作用(转)
    MySQL中插入相关
    MyBatis 中错误信息详情、原因分析及解决方案
    Java 的全限定类名
  • 原文地址:https://www.cnblogs.com/ningxu/p/3393168.html
Copyright © 2020-2023  润新知