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

通过反射来获取某一个类的构造器:
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);
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一、概述 Java反射机制定义 Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类中的所有属性和方法...
    CoderZS阅读 5,536评论 0 26
  • 一:java概述: 1,JDK:Java Development Kit,java的开发和运行环境,java的开发...
    慕容小伟阅读 5,801评论 0 10
  • 整理来自互联网 1,JDK:Java Development Kit,java的开发和运行环境,java的开发工具...
    Ncompass阅读 5,423评论 0 6
  • 喜乐操练90天——43《神秘的朋友》 你使他有洪福,直到永远,又使他在你面前欢喜快乐。(诗篇21:6) 喜...
    午后的咖啡与茶阅读 2,118评论 0 0
  • 文/清风远行 深夜凌晨一点钟,我坐在书桌的电脑前发呆,桌上的烟灰缸里已经堆满了烟头。看着朋友圈里的照片你靠在一个男...
    怪马m阅读 5,501评论 2 7