Java反射机制

来源:互联网 发布:mac后台程序关不上 编辑:程序博客网 时间:2024/06/09 14:46

Java反射机制

一、什么是反射机制
        简单的来说,反射机制指的是程序在运行时能够获取自身的信息。在java中,只要给定类的名字,
    那么就可以通过反射机制来获得类的所有信息。
二、哪里用到反射机制
        有些时候,我们用过一些知识,但是并不知道它的专业术语是什么,在刚刚学jdbc时用过一行代码,
    Class.forName("com.mysql.jdbc.Driver.class").newInstance();但是那时候只知道那行代码是生成
    驱动对象实例,并不知道它的具体含义。听了反射机制这节课后,才知道,原来这就是反射,现在很多开
    框架都用到反射机制,hibernate、struts都是用反射机制实现的。
三、反射机制的优点与缺点
        为什么要用反射机制?直接创建对象不就可以了吗,这就涉及到了动态与静态的概念,
    静态编译:在编译时确定类型,绑定对象,即通过。
    动态编译:运行时确定类型,绑定对象。动态编译最大限度发挥了java的灵活性,体现了多
    态的应用,有以降低类之间的藕合性。
    一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中
    它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编
    译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如
    这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能
    的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功
    能。
       它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它
    满足我们的要求。这类操作总是慢于只直接执行相同的操作。
四、利用反射机制能获得什么信息
         一句话,类中有什么信息,它就可以获得什么信息,不过前提是得知道类的名字,要不就没有后文了
    首先得根据传入的类的全名来创建Class对象。
    Class c=Class.forName("className");注明:className必须为全名,也就是得包含包名,比如,cn.netjava.pojo.UserInfo;
    Object obj=c.newInstance();//创建对象的实例
    OK,有了对象就什么都好办了,想要什么信息就有什么信息了。  
    获得构造函数的方法
    Constructor getConstructor(Class[] params)//根据指定参数获得public构造器

    Constructor[] getConstructors()//获得public的所有构造器

    Constructor getDeclaredConstructor(Class[] params)//根据指定参数获得public和非public的构造器

    Constructor[] getDeclaredConstructors()//获得public的所有构造器
    获得类方法的方法
    Method getMethod(String name, Class[] params),根据方法名,参数类型获得方法

    Method[] getMethods()//获得所有的public方法

    Method getDeclaredMethod(String name, Class[] params)//根据方法名和参数类型,获得public和非public的方法

    Method[] getDeclaredMethods()//获得所以的public和非public方法
    获得类中属性的方法
    Field getField(String name)//根据变量名得到相应的public变量

    Field[] getFields()//获得类中所以public的方法

    Field getDeclaredField(String name)//根据方法名获得public和非public变量

    Field[] getDeclaredFields()//获得类中所有的public和非public方法
    常用的就这些,知道这些,其他的都好办……
五、用反射机制能干什么事
        刚开始在使用jdbc时侯,在编写访问数据库时写到想吐,有八个表,每个表都有增删改查中操作
    那时候还不知道有反射机制这个概念,所以就对不同的表创建不同的dao类,这样不仅开发速率地,而且代码
    冗余的厉害,最要命的是看着差不多的,然后直接复制修改,由于容易犯各种低级的错误(大小写啊,多一
    个或少一个字母啊……),一个错误就可以让你找半天。
        有了java反射机制,什么都好办了,只需要写一个dao类,四个方法,增删改查,传入不同的对象,就OK啦,
    无需为每一个表都创建dao类,反射机制会自动帮我们完成剩下的事情,这就是它的好处。说白了,反射机制就是专门
    帮我们做那些重复的有规则的事情,所以现在很多的自动生成代码的软件就是运用反射机制来完成的,只要你按照规则

    输入相关的参数

六、举例

1)、需要获得java类的各个组成部分,首先需要获得类的Class对象,获得Class对象的三种方式:

       Class.forName(classname)   用于做类加载

       obj.getClass()                          用于获得对象的类型

       类名.class                      用于获得指定的类型,传参用

 

2)、反射类的成员方法:

       Class clazz = Person.class;

       Method method =clazz.getMethod(methodName, new Class[]{paramClazz1, paramClazz2});

       method.invoke();

      

3)、反射类的构造函数:

       Constructor con =clazz.getConstructor(new Class[]{paramClazz1, paramClazz2,...})

       con.newInstance(params...)

 

4)、反射类的属性:

       Field field = clazz.getField(fieldName);

       field.setAccessible(true);

       field.setObject(value);


例一

public class RefilectDemo03 {
    public static void main(String[] args) throws Exception {
        //获得指定类的对象                                                                                                                    第一步
        Class<?> c1=Class.forName("fanshe001.Rect");
        //构建类对象的实例
        Rect rect=(Rect) c1.newInstance();//(默认为无参的构造类型)      第二步               
       


例二

        //获得类中的构造函数   某一个构造函数
        Constructor<?> c=c1.getDeclaredConstructor(int.class,int.class);     //第一步
        //通过构造函数对象构建累的对象
        Rect tect2=(Rect)c.newInstance(10,20);                               //第二步
        System.out.println("tect2="+tect2);    
        //反射调用对象方法
        Method m=c1.getDeclaredMethod("getArea");                             //第三步
        Object result=m.invoke(tect2);                                        //第四步
        System.out.println(result);
    }

例三


public class ReflectDemo04 {
        public static void main(String[] args) throws Exception, SecurityException {
            List<String> list=new ArrayList<String>();
            list.add("A");
            list.add("B");
            list.add("100");
            System.out.println("list2="+list);
            //用反射象集合写入100                 
            Class<?> c=list.getClass();                    //第一步
            //获得类对象的已添加方法
        Method    m=c.getDeclaredMethod("add",Object.class );      //第二步
            m.invoke(list, 100);                                 //第三步
            System.out.println("list="+list);
            
    }
}

例四

private static void doMethods() throws Exception{
    //这个类是获取某个方法并给方法中的对象赋值
    //反射应用的起点首先要获得的类对象  每个类对应的字节码对象只有一个                                                                第一步
    Class<?> c1=Point.class;
    Method ms[]=c1.getDeclaredMethods();//获得所有的方法                                                                第二步
    for(Method m:ms){
        System.out.println(m.getName());//获取所有方法名
    }
    Point p1=new Point();
    Method m1=c1.getDeclaredMethod("setY", int.class);//获得set方法
    m1.invoke(p1, 20);//给m1赋值  更改p1的属性值                                                                                            第三步   
    Method m=c1.getDeclaredMethod("getY");//获得类中指定的方法 //获得动态m方法
    //执行point 对象的getY方法(动态过程)
    Object result=m.invoke(p1);//调用实力方法执行业务                                                                              第四步
    System.out.println("result2="+result);//为0
}