美文网首页java入门基础
反射机制(三)操作构造器

反射机制(三)操作构造器

作者: 无关风月oo | 来源:发表于2018-06-05 02:59 被阅读15次

    通过反射来获取某一个类的构造器:
    1>获取该类的字节码对象;
    2>从该字节码对象中去找需要获取的构造器;


    Class类获取构造器方法:

    Constructor类:
    --表示类中构造器的类型,Constructor的实例就是某一个类中的某一个构造器;

    public Constructor<?>[] getConstructors() ;
    --该方法只能获取当前Class所表示类的public修饰的构造器;
    
    public Constructor<?>[] getDeclaredConstructors();
    --获取当前Class所表示类的所有构造器,和访问权限无关;
    
    public Constructor<T> getConstructor(Class<?>... parameterTypes);
    --获取当前Class所表示类中指定的一个public的构造器
    
    public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes);
    --获取当前Class所表示类中指定的一个构造器;
    
    参数:parameterTypes
    --表示:构造器参数的Class类型;
    

    
    public class _0_GetConstructor {
        public static void main(String[] args) throws Exception{
    //        获取所有的构造器
            getAlls();
    //        获取指定的一个构造器
            getAppoint();
        }
        private static void getAlls() {
    //        1.获取构造器所在类的字节码对象
            Class clz = User.class;
    //        2.获取clz对象中所有的构造器
    //        该方法只能获取当前Class所表示类的public修饰的构造器;
            Constructor<User>[] cons = clz.getConstructors();
            System.out.println(cons.length);//2
            for (Constructor c : cons) {
                System.out.println(c);
                //public com.reflection._1_constructor.User()
                //public com.reflection._1_constructor.User(java.lang.String)
            }
            System.out.println("------------------------------");
    //        获取当前Class所表示类的所有构造器,和访问权限无关;
            cons = clz.getDeclaredConstructors();
            System.out.println(cons.length);//3
            for (Constructor c : cons) {
                System.out.println(c);
                //public com.reflection._1_constructor.User()
                //public com.reflection._1_constructor.User(java.lang.String)
                //private com.reflection._1_constructor.User(java.lang.String,int)
            }
        }
    
        public static void getAppoint() throws Exception {
            System.out.println("------------------------------");
    //        1.获取构造器所在类的字节码对象
            Class clz = User.class;
    //        需求1:获取public User()
            Constructor<User> con = clz.getConstructor();
            System.out.println(con);//public com.reflection._1_constructor.User()
    
    //        需求2:获取public User(String name)
            con = clz.getConstructor(String.class);
            System.out.println(con);
    
    //        需求3:获取public User(String name ,int age)
    //        con = clz.getConstructor(String.class,int.class);//报错:Exception in thread "main" java.lang.NoSuchMethodException:
    //                                                                com.reflection._1_constructor.User.<init>(java.lang.String, int)
            con = clz.getDeclaredConstructor(String.class,int.class);
            System.out.println(con);
        }
    }
    class User{
        public User(){}
    
        public User(String name){}
    
        private User(String name ,int age){}
    }
    

    构造器最大的作用:创建对象;
    为什么使用反射创建对象而不直接使用new呢?
    在框架中,提供给我们的都是字符串;
    很多框架(比如Spring)都是配置化的(比如通过XML文件配置JavaBean,Action之类的),为了保证框架的通用性,它们可能需要根据配置文件加载不同的对象或类,调用不同的方法,这个时候就必须用到反射——运行时动态加载需要加载的对象。


    使用反射创建对象

    步骤:
    1.找到构造器所在类的字节码对象;
    2.获取构造器对象;
    3.使用反射,创建对象;


    Constructor<T>类:表示类中构造器的类型,Constructor的实例就是某一个类中的某一个构造器;

    常用方法:

    public T newInstance(Object... initarges);
    -若调用带参数的构造器,只能使用该方式;
    -参数:initarges,表示调动构造器的实际参数;
    -返回:返回创建的实例,T表示Class所表示类的类              型;
    
    如果:一个类中的构造器是外界可以    直接访问的,同时没有参数,那么可以直接使用Class类中的newInstance()方法创建对象;
    
    -public Object newInstance();//相当于new 类名();
    

    访问私有的成员

    -(Constructor、Method、Field):

    必须先设置可访问的:

    语法:对象.setAccessible(true);
    见源码:
    -public final class Constructor<T> extends Executable ...
    -public abstract class Executable extends AccessibleObject{
     public static void setAccessible(AccessibleObject[] array, 
     boolean  flag)
     }
    

    //使用反射调用构造器所在类来创建其对象
    public class _1_CreateObject {
        public static void main(String[] args) throws Exception{
            Class<Person> clz = Person.class;
            Person p = clz.newInstance();
            Constructor<Person> con = clz.getConstructor();
            Person instance = con.newInstance();//调用无参数构造器
    
            con = clz.getConstructor(String.class);
            con.newInstance("INSOMNIA");
    
            con = clz.getDeclaredConstructor(String.class,int.class);
    //        con.newInstance("INSOMNIA",18);//报错,原因:
    //Exception in thread "main" java.lang.IllegalAccessException://非法访问异常
    //Class com.reflection._1_constructor._1_CreateObject
    //can not access a member of class com.reflection._1_constructor.Person with modifiers "private"
            /** 设置当前构造器可以访问*/
            con.setAccessible(true);//可解决上述问题;使用枚举,即使反射也不行;
            con.newInstance("INSOMNIA",18);
        }
    }
    class Person{
        public Person(){
            System.out.println("调用无参数构造器");
        }
    
        public Person(String name){
            System.out.println("调用有参数构造器"+name);
        }
    
        private Person(String name ,int age){
            System.out.println("调用有参数构造器"+"name = "+name+" & "+"age = "+age);
        }
    }
    

    相关文章

      网友评论

        本文标题:反射机制(三)操作构造器

        本文链接:https://www.haomeiwen.com/subject/gxmjsftx.html