Java 注解和反射(五)创建运行时类的对象

获取运行时类的完整结构

通过反射获取运行时类的完整结构

Field,Method,Constructor,Superclass,Interface,Annotation

~实现的全部接口

~所继承的父类

~全部的构造器

~全部的方法

~全部的Field

~注解

............

//获得类的信息
public class Test07 {
    public static void main(String[] args) throws ClassNotFoundException {
        Class c1 = Class.forName("refection.User");
        //获得类的名字
        System.out.println(c1.getName());//包名+类名
        System.out.println(c1.getSimpleName());//类名

        //获得类的属性
        Field[] fields = c1.getFields();//只能找到public属性的
        fields = c1.getDeclaredFields();//找到全部属性
        for (Field field: fields) {
            System.out.println(field);
        }

        //获得类的方法
        Method[] methods = c1.getMethods();//获得本类及其父类的public所有方法
        methods = c1.getDeclaredMethods();//获得本类的方法
        for (Method method: methods) {
            System.out.println(method);
        }

        //获得指定构造器
        System.out.println("==============");
        Constructor[] constructors = c1.getConstructors();//获得public
        constructors = c1.getDeclaredConstructors();//获得全部
        for (Constructor constructor: constructors) {
            System.out.println(constructor);
        }
    }
}

 

动态创建对象

创建类的对象:调用Class对象的newInstance()方法

  1.类必须要有一个无参数构造器

  2.类的构造器的访问权限需要足够

思考?==》难道没有无参的构造器就不能创建对象了吗?只要在操作的时候明确的调用类中的构造器,

     并将参数传递下去之后,才可以实例化操作

步骤如下:

    1.通过Class类的getDeclaredConstructor(Class.....parameterTypes)取得本类的指定形参类型的构造器

    2.向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数

    3.通过Constructor实例化对象

 

调用指定方法

通过反射,调用类中的方法,通过Method类完成

  1.通过Class类的getMethod(String name,Class....parameterTypes)方法取得一个Mehod对象并设置此方法操作时所需要的参数类型

  2.之后使用Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息

Object invoke(Object obj,Object[] args)

``Object对应原方法的返回值,若原方法无返回值,此时返回null

``若原方法若为静态方法,此时形参Object obj可谓null

``若原方法形参列表为空,则Object[] args 为null

``若原方法声明为private,则需要在调用此invoke()方法前,显式调用方法对象的setAccessible(true)方法,将能访问private方法

setAccessible

··Method和Field,Constructor对象都有setAccessible()方法

··setAccessible作用是启动和禁用访问安全检查的开关

··参数值为true则指示反射的对象在使用时应该取消Java语言访问检查

  1.提高反射效率。如果代码中必须使用反射,而该句代码需要频繁被调用,那么请设置true

  2.使得原本无法访问的私有成员也可访问

··参数值为false则指示反射的对象应该实施Java语言访问检查

//动态创建对象,通过反射
public class Test08 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得Class对象
        Class c1 = Class.forName("refection.User");
        //构造一个对象
        User user = (User) c1.newInstance();//本质调用无参构造器
        System.out.println(user);

        //通过构造器创建对象
        Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) constructor.newInstance("呜呜", 001, 18);
        System.out.println(user2);

        //通过反射调用普通方法
        User user3 = (User) c1.newInstance();
        //通过反射获取一个方法
        Method setName = c1.getDeclaredMethod("setName", String.class);
        //激活(对象,“方法值”)
        setName.invoke(user3,"小磊");
        System.out.println(user3.getName());

        //通过反射操作属性
        User user4 = (User) c1.newInstance();
        Field name = c1.getDeclaredField("name");
        //不能直接操作私有属性,需要关闭程序安全检测
        name.setAccessible(true);//设置true
        name.set(user4,"小朱");
        System.out.println(user4.getName());
    }
}

 其中的User类见Java注解和反射(三),根据需要删除无参构造器

 

分析性能问题

//分析性能问题
public class Test09 {
    //普通方式调用
    public static void test01(){
        User user = new User();
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            user.getName();
        }
        long endTime = System.currentTimeMillis();
        System.out.println("普通方式执行十亿次="+(endTime-startTime)+"ms");
    }

    //反射方法调用
    public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();
        Method getName = c1.getDeclaredMethod("getName", null);
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("反射方法执行十亿次="+(endTime-startTime)+"ms");
    }
    //反射方式调用,关闭检测
    public static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();
        Method getName = c1.getDeclaredMethod("getName", null);
        getName.setAccessible(true);
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("反射方式调用,关闭检测执行十亿次="+(endTime-startTime)+"ms");
    }

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
        test01();
        test02();
        test03();
    }
}

 

 

 

 其中的User类见Java注解和反射(三)

 

posted on 2022-11-29 10:28  键盘敲烂的朱  阅读(54)  评论(0编辑  收藏  举报