创建运行时类的对象

文章详细介绍了Java中的反射机制,包括如何获取类的完整结构,如类名、属性、方法、构造器以及注解。通过Class对象,可以访问到类的public成员和所有成员,以及动态创建对象和调用方法。此外,还讨论了setAccessible(true)方法在访问私有成员时的作用,并进行了性能对比分析,展示了反射调用的效率差异。
摘要由CSDN通过智能技术生成

获取运行时类的完整结构

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

Field、Method、Constructor、Superclass、Interface、Annotation

  • 实现的全部接口
  • 所继承的父类
  • 全部的构造器
  • 全部的方法
  • 全部的Field
  • 注解......
     
package reflection;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

//获得类的信息
public class Test07 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {
        Class c1 = Class.forName("reflection.User");
        //获得类的名字
        System.out.println(c1.getName());//获取类的名字 包名+类名
        System.out.println(c1.getSimpleName());//获取类的简单名字  类名
        
        //获得类的属性
        Field[] fields01 = c1.getFields();//getFields() 只能找到public属性
        for (Field field : fields01) {
            System.out.println(field);//打印不出来
        }
        Field[] fields02 = c1.getDeclaredFields();//getDeclaredFields()可以找到所有属性
        for (Field field : fields02) {
            System.out.println(field);//可以打印出来
        }
        
        //获得指定属性的值
        Field name = c1.getDeclaredField("name");
        System.out.println(name);

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

        Method[] methods02 = c1.getDeclaredMethods();//获得本类的所有方法
        for (Method method : methods02) {
            System.out.println("getDeclaredMethods:    "+method);
        }

        //获得指定的方法
        Method getName = c1.getMethod("getName",null);//名字+参数
        Method setName = c1.getMethod("setName",String.class);//名字+参数
        System.out.println(getName);
        System.out.println(setName);

        //获得构造器
        Constructor[] constructors01 = c1.getConstructors();//获得本类的所有public构造器
        for (Constructor constructor : constructors01) {
            System.out.println("getConstructors:    "+constructor);
        }
        Constructor[] constructors02 = c1.getDeclaredConstructors();//获得本类的所有构造器
        for (Constructor constructor : constructors02) {
            System.out.println("getDeclaredConstructors:     "+constructor);
        }

        //获得指定的构造器
        Constructor constructor = c1.getConstructor(String.class, int.class, int.class);
        System.out.println(constructor);
        Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        System.out.println(declaredConstructor);


    }
}

动态创建对象执行方法

有了Class对象,能做什么?

 创建类的对象:调用Class对象的newlnstance()方法
        1)类必须有一个无参数的构造器
        2)  类的构造器的访问权限需要足够
思考?

        难道没有无参的构造器就不能创建对象了吗?只要在操作的时候明确的调用类中的构造器,并将参数传递进去之后,才可以实例化操作
步骤如下:
        1)通过Class类的getDeclaredConstructor(Class ... parameterTypes)取得本类的指定形参类型的构造器
        2)向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数

        3)通过Constructor实例化对象

//通过构造器创建对象
        Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) declaredConstructor.newInstance("李星辰",0210, 18);
        System.out.println(user2);

调用指定的方法:

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

  1. 通过Class类的getMethod(String name,Class...parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型
  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的方法
package reflection;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

//动态的创建对象,通过反射
public class Test08 {
    public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得Class对象
        Class c1 = Class.forName("reflection.User");

        //构造一个对象
        User user = (User)c1.newInstance();//本质是调用了类的无参构造器 等号右边(User)表示强制转换成了User类
        System.out.println(user);//当类里没有无参构造器就会出错

        //通过构造器创建对象
        Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) declaredConstructor.newInstance("李星辰",0210, 18);
        System.out.println(user2);

        //通过反射调用普通方法

        User user3 = (User)c1.newInstance();
        //通过反射获取一个方法
        Method setName = c1.getDeclaredMethod("setName", String.class);
        //invoke:激活的意思
        setName.invoke(user3,"星辰"); //(对象,“方法的值”)
        System.out.println(user3.getName());

        //通过反射操作属性
        User user4 = (User)c1.newInstance();
        Field name = c1.getDeclaredField("name");

        //当属性为priva私有时,不能直接操作,此时需要关闭程序的安全监测,可以通过属性或方法的setAccessible(true)关闭
        //name.setAccessible(true);
        name.set(user4,"辰星");
        System.out.println(user4.getName());


    }
}

setAccessible :

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

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

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

  • 提高反射的效率。如果代码中必须用反射,而该句代码需要频繁的被调用,那么请设置为true
  • 使得原本无法访问的私有成员也可以访问

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

性能对比分析 

测试普通方式调用时间、反射方式调用时间、反射方式并关闭检测调用时间

package reflection;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

//分析性能问题
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();//结束时间
        long time = endTime-startTime;
        System.out.println("普通方式执行十亿次需要"+time+"ms");
    }

    //反射方式调用
    public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();

        Method declaredMethod = c1.getDeclaredMethod("getName",null);

        long startTime = System.currentTimeMillis();//开始时间
        for (int i = 0; i < 1000000000; i++) {
            declaredMethod.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();//结束时间
        long time = endTime-startTime;
        System.out.println("普通方式执行十亿次需要"+time+"ms");
    }

    //反射方式调用 关闭检测
    public static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();

        Method declaredMethod = c1.getDeclaredMethod("getName",null);
        declaredMethod.setAccessible(true);//关闭检测

        long startTime = System.currentTimeMillis();//开始时间
        for (int i = 0; i < 1000000000; i++) {
            declaredMethod.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();//结束时间
        long time = endTime-startTime;
        System.out.println("普通方式执行十亿次需要"+time+"ms");
    }

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

}

 

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值