注解与反射04

注解与反射04

4.创建运行时类的对象

获取运行时类的完整结构

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

Field、Method、Constructor、SuperClass、Interface、Annotation

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

代码实战

// 获取类的信息
public class Test08 {

    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {

        Class c1 = Class.forName("com.qinggangqiang.reflection.User");

        // 获取类的名字
        System.out.println(c1.getName());// 获得包名 + 类名
        System.out.println(c1.getSimpleName());// 获得类名

        // 获得类的属性
        System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
        Field[] fields = c1.getFields();
        for (Field field : fields) {
            System.out.println("c1.getFields() 的值:" + field);// 执行结果:没有值!原因是:属性值类型是私有的!
        }

        Field[] declaredFields = c1.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println("declaredField:" + declaredField);//执行结果:有值
        }

        // 总结上述获取字段:c1.getFields();只能找到public属性;c1.getDeclaredFields();能找到全部属性

        // 获得指定属性的值
//        Field name = c1.getField("name");
//        System.out.println(name);// 执行结果:报错!原因是使用只能获取公有属性方法的方法,会发生错误,因为name的属性是私有的!
        Field name1 = c1.getDeclaredField("name");
        System.out.println(name1);

        // 获得类的方法
        System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
        Method[] methods = c1.getMethods();
        for (Method method : methods) {
            System.out.println("正常的:" + method);//获得本类及其父类的全部public方法
        }
        Method[] declaredMethods = c1.getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            System.out.println("@getDeclaredMethods():" + declaredMethod);// 只获得本类的所有方法
        }

        // 获得指定方法
        // 思考:为什么有参数值?因为重载!仅仅方法名称是不唯一的,参数的不同才会执行不同的方法
        Method getName = c1.getMethod("getName", null);
        Method setName = c1.getMethod("setName", String.class);
        System.out.println(getName);
        System.out.println(setName);


        // 获得指定的构造器
        System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
        Constructor[] constructors = c1.getConstructors();//获得public的方法
        for (Constructor constructor : constructors) {
            System.out.println("constructor:" + constructor);
        }
        Constructor[] declaredConstructors = c1.getDeclaredConstructors();//获得本类的全部方法
        for (Constructor declaredConstructor : declaredConstructors) {
            System.out.println("declaredConstructor:" + declaredConstructor);
        }

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


}

小结

  • 在实际的操作中,取得类的信息的操作代码,并不会经常开发。
  • 一定要熟悉java.lang.reflect的作用,反射机制。
  • 如何取得属性、方法、构造器的名称,修饰符等。

思考1:有了Class对象,能做什么?

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

思考2:难道没有无参的构造器就不能创建对象了吗?

只要在操作的时候明确调用类中的构造器,并将参数传递进去之后,才可以实例化操作。

步骤如下:

  1. 通过Class 类的getDeclaredConstructor(Class ... parameterType) 取得本类的指定形参类型的构造器
  2. 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数。
  3. 通过Constructor实例化对象

代码实现

// 动态的创建对象,通过反射
public class Test09 {

    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {

        // 获得Class对象
        Class c1 = Class.forName("com.qinggangqiang.reflection.User");

        // 构造一个对象
        Object o = c1.newInstance();// 本质是调用了类的无参构造器
        User user = (User) o;
        System.out.println(user);// User(name=null, id=0, age=0)

        // 通过构造器创建对象
        Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) constructor.newInstance("倾·浪漫老爷#Yolo-Qing", 001, 18);
        System.out.println(user2);// User(name"倾·浪漫老爷#Yolo-Qing", id=1, age=18)


        // 通过反射调用普通方法
        User user3 = (User) c1.newInstance();
        // 通过反射获取一个方法
        Method setName = c1.getDeclaredMethod("setName", String.class);

        // invoke:激活的意思
        // 使用:(对象,"方法的值")
        setName.invoke(user3,"倾·浪漫老爷#Yolo-Qing");
        System.out.println(user3.getName());// 倾·浪漫老爷#Yolo-Qing

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

        // 操作
//        name.set(user4,">>>>>>>>>>>倾·浪漫老爷#Yolo-Qing>>>>>>>>>>>>>");
//        System.out.println(user4.getName());// 执行结果:报错!原因是:属性类型是私有属性!

        // 但是通过反射可以进一步获取私有属性的值!
        // 设置:关闭权限检测
        name.setAccessible(true);
        name.set(user4,">>>>>>>>>>>倾·浪漫老爷#Yolo-Qing>>>>>>>>>>>>>");
        System.out.println(user4.getName());//>>>>>>>>>>>倾·浪漫老爷#Yolo-Qing>>>>>>>>>>>>>

        // 总结:不能直接操作私有属性,我们需要关闭程序的安全检测,属性或者方法的setAccessible(true); 

    }

}

调用指定的方法

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

  • 通过Class类的getMethod(String name,Class ... parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型。
  • 之后使用Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的Obj 对象的参数信息。

在这里插入图片描述

Object invoke(Object obj,Object[] args):
1. Object 对应原方法的返回值,若原方法无返回值,此时返回null
2. 若原方法为静态方法,此时形参Object obj 可为 null
3. 若原方法形参列表为空,则 Object[] args 为 null
4. 若原方法声明为private,则需要在调用此invoke() 方法前,显示调用方法对象的 setAccessible(true)方法,将可访问private 的方法。

setAccessible(true)

1.Method 和 Field、Constructor对象都有 setAccessible()方法。
2.setAccessible 作用是启动和禁用访问安全检查的开关。
3.参数值为true 则指示反射的对象在使用时应该取消Java语言访问检查。
	(1)提高反射的效率。如果代码中必须使用反射,而该句代码需要频繁的被调用时,那么请设置为true。
	(2)使得原本无法访问的私有成员也可以访问。
4.参数值为false 则指示反射的对象应该实施Java语言访问检查。
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值