Java 反射(Reflection)

617 查看

反射的核心就是Class对象,每一个类被jvm加载都会有一个对应的class对象,这个class对象包含了这个类的结构信息,反射就是会通过反射api反复操作这个class对象(属性,方法,注解,构造器,泛型),但是反射会降低程序的运行效率,比普通方法要慢30倍,通过setAccessble(true) 跳过安全检查(可以读取private 的属性),提高4倍的程序运行效率
反射将普遍运用到各个类型的框架。
效率:
1,运行效率:使用反射 减低程序的运行效率。
2,开发效率:由于反射运用到各个java框架中,大大加快了开发的效率。

下面就是一些简单的例子来具体阐述反射的一些特点:

 package com.mk;
     public class Test{
     public static void main(String [] args){
        Foo foo = new Foo();//foo 表示foo的实例对象
        //任何一个类都是class的实例对象,有3钟表达方式
        //1,任何一个类都有一个隐含的成员变量Class
        Class c1 = Foo.class;
        //2,通过getClass方法获得
        Class c2 = foo.getClass();
        //不管是c1还是c2都代表Foo类的类类型,一个类只能是Class的实例对象
        System.out.println(c1==c2);//true
        //3,第三种表达方式
        Class c3 = null;
        try {
            c3 = Class.forName("com.mk.Foo");//类的全称
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        System.out.println(c2==c3);//true

        //可以通过类的类类型创建该类的对象实例
        try {
            Foo f1 = (Foo) c1.newInstance();//需要无参数的构造方法
            f1.print();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
     }
    }
       class Foo{
        void print(){
        System.out.println("print parent");
       }
    }

动态加载类
new 创建对象 是静态的加载类,在编译的时刻就需要加载 所有可能使用到的类
通过动态的加载类可以解决这个问题
运行时刻加载 就是想用哪个就加载哪个,不用就不加载。
通过反射api 操作类的方法

method.invoke(对象,参数列表)
    package com.mk;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    public class Test1 {
    public static void main(String[]args) throws
      NoSuchMethodException,      
        InvocationTargetException,
     IllegalAccessException {
        A a1 = new A();
        Class c = a1.getClass();//得到类类型
        Method m = c.getMethod("print",int.class,int.class);
        //方法的反射操作
        //a1.print(10,20);方法的反射操作是用m对象进行方法的调用,和a1.print()的调用效果相同
        //方法如果没有返回值返回null,如果有返回值就返回具体的值
        //Object obj = m.invoke(a1,new Object[]{10,20});
        //或者有几个参数传几个参数
        Object ob = m.invoke(a1,10,20);
      }
     }
    class A{
    public void print(int a,int b){
        System.out.println(a+b);
      }
    }

通过class,method来认识泛型的本质

 package com.mk;
    java.lang.reflect.Method;
    java.util.ArrayList;
    class test2 {
    public static void main(String[] arg){
        ArrayList list = new ArrayList();
        ArrayList<String> ls = new ArrayList<String>();
        ls.add("hello");
        //ls.add(20);//泛型检查错误,20是加不进去的
        Class c1 = list.getClass();
        Class c2 = ls.getClass();
        System.out.println(c1==c2);//反射的操作都是编译之后的操作
        /**
         * c1==c2 结果返回true说明编译之后的集合的泛型是去泛型化的,java中
         * 集合的泛型,是防止错误输入的,只有在编译阶段是有效的,在其他的阶段都是
         * 无效的
         * 可以通过方法的反射来操作 绕过编译
         */
          try {
            Method m = c2.getMethod("add",Object.class);
            m.invoke(ls,100);
            System.out.println(ls.size());
            System.out.println(ls);//[hello, 100] 结果int 是可以加进去的
         } catch (Exception e) {
            e.printStackTrace();
        }

      }
    }

如何通过反射api 操作属性,构造器

public class User {
    private int id;
    private String name;
    //javabean 无参构造器,用于java的反射机制的初始化
    public User() {
        
    }
    public User(int id, String name) {
        super();
        this.id = id;
        this.name = name;
    }
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}
        Class clazz 
             = Class.forName("com.test.reflection.Bean.User");
        //获取名字
        System.out.println(clazz.getName());//包名+类名
        System.out.println(clazz.getSimpleName());
        //获取属性方法(重载,根据参数找方法)
        Field [] fs = clazz.getDeclaredFields();
        System.out.println(fs.length);//获取所有的fields,不论public还是private
        Field [] field = clazz.getFields();//获取公共属性
        for(Field fields:fs){
            System.out.println("属性:"+fields);
        }
        //获取方法信息
        Method [] method 
             = clazz.getDeclaredMethods();//获取所有的方法
        Method m1 = 
                clazz.getDeclaredMethod("getName",null);
        for(Method m:method){
            System.out.println("方法:"+m);
        }
        //获取构造器
        Constructor c1 = clazz.getConstructor(null);
        System.out.println("无参构造器:"+c1);
        Constructor c2 = clazz.getConstructor(int.class,String.class);
        System.out.println("有参构造器:"+c2);
        Constructor [] constructors 
            = clazz.getDeclaredConstructors();
        for(Constructor c:constructors){
            System.out.println("构造器:"+c);
        }
        
        
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
}

 //动态操作构造器
      try {
        //通过反射api调用构造方法
        User user = (User) clazz.newInstance();//其实是调用了user的无参构造方法
        System.out.println(user);
        //调用有参构造器
        Constructor constructor 
            = clazz.getDeclaredConstructor(int.class,String.class);
        User u 
            = (User) constructor.newInstance(12,"michael");
        System.out.println(u.getName());
        //通过反射api调用普通的方法
        User u1 
             = (User)clazz.newInstance();
        Method method 
             = clazz.getDeclaredMethod("setName", String.class);
        method.invoke(u1, "mike");//setName("mike");
        System.out.println(u1.getName());
        //通过反射操作属性
        User u2 = (User) clazz.newInstance();
        Field  f 
             = clazz.getDeclaredField("name");//属性必须是public的
        f.setAccessible(true);//不需要安全检查,直接访问
        f.set(u2, "mike2");
        System.out.println(u2.getName());//通过反射直接读动态值
        System.out.println(f.get(u2));
    } catch (Exception e) {
        e.printStackTrace();
    }