• 2022-08-10 第四小组 修身课 学习笔记(every day)


    注解和反射


    注解:Annotation,Java标注,JDK5引入的一种机制。
    Java中类,方法,变量,参数,包都可以被标注
    元注解:专门给注解加的注解
    我们发现注解中可以有方法,
    1、定义方法的格式:String name();
    2、可以有默认值,也可以没有,如果没有默认值在使用的时候必须填写对应的值
        如果需要有默认值,使用default指定默认值。
    3、如果想在使用的时候不指定具体的名字,
    如果不学习反射,注解没啥用!!!
    在java的整个的注解体系中,有3个非常重要的主干类,
    1、Annotation 接口,定义一些常用的方法
    2、ElementType 枚举
        它用来指定注解的类型。说人话,我的注解要用在哪里???
    3、RetentionPolicy 枚举
        它用来指定注解的策略。不同类型的策略指定的注解的作用域不同。
       (1)SOURCE,注解仅存在于编译器处理期间,编译期处理完之后,这个注解就没用了
       (2)CLASS,注解在.class文件中依然有效。
       (3)RUNTIME,编译期是不起作用的,只有在运行期才由JVM读取。
    Java自带的注解,10个。4个注解java.lang.annotation
                         6个注解在java.lang
    作用在代码上的注解
    1、@Override,检查该方法是否是重写方法。如果返现其父类,或者是引用的接口中没有该方法,会报错
    2、@Deprecated,标记的方法,过时的方法。
    3、@SuppressWarnings编辑器去忽略警告
    4、@SafeVarargs,JDK7支持忽略任何使用参数为泛型变量的方法或构造器产生的警告
    5、@FunctionalInterface,JDK8开始支持,表示一个接口为函数式接口
    6、@Repeatable,JDK8开始支持,标识某个注解可以在同一个声明上使用多次
        all:忽略所有警告
        boxing:忽略装箱、拆箱警告
        rawtypes:使用生成时没有指定数据类型
        unchecked:忽略没有进行类型检查操作的警告
        unused:忽略没有使用的警告
    元注解:
    1、@Retention:标识这个注解作用域
    2、@Documented:标记这个注解是否包含在用户文档中
    3、@Target:这个注解可以修饰哪些信息
    4、@Inherited:如果一个类用上了@Inherited注解,那么其子类也会继承这个注解

    反射机制

            通过java语言中的反射机制可以操作字节码文件

    反射机制相关的类在那个包下?

            java.lang.reflect.*

    反射机制相关重要的类

            java.lang.Class 代表整个字节码,代表一个类型,代表整个类

            java.lang.reflect.Method: 代表字节码中的方法字节码,代表方法

            java.lang.reflect.Constructor:代表字节码中的构造方法字节码 ,代码构造器

            java.lang.reflect.Field: 代表字节码中的属性字节码,代表静态变量


    使用反射创建对象

    反射创建对象的几种方式

    Class.forname()

            1.静态方法

            2.方法的参数是一个字符串

            3.字符串需要一个完整的类名

            4.完整类名必须带有包名。java.lang包也不能省略 

    1. /*获得一个类的字节码*/
    2. Class pClass = Class.forName("com.bjsxt.demo1.Person");
    3. /*
    4. * 获得类的构造方法
    5. * */
    6. Constructor cons = pClass.getDeclaredConstructor(int.class, String.class, String.class);
    7. /*
    8. * 让构造方法执行
    9. * */
    10. Person person=(Person) cons.newInstance(1,"小明","男");
    11. System.out.println(person);
    12. /*
    13. * 可以用字节码直接实例化该类的一个对象
    14. * 字节码.newInstance() 其实就是在调用类的无参数构造方法实例化一个对象
    15. * 如果使用该方式,必须保证类中必须有空参数构造方法
    16. * */
    17. Person person2 = (Person)pClass.newInstance();
    18. System.out.println(person2)

    反射给对象属性赋值

    1. /*获得一个类的字节码*/
    2. Class pClass = Class.forName("com.bjsxt.demo1.Person");
    3. Person person = (Person)pClass.newInstance();
    4. System.out.println(person);
    5. /*
    6. * 通过field对象完成给对象属性赋值
    7. * */
    8. Field id = pClass.getDeclaredField("id");
    9. Field lastname = pClass.getDeclaredField("lastname");
    10. Field gender = pClass.getDeclaredField("gender");
    11. Field firstname = pClass.getDeclaredField("firstname");
    12. // private修饰的属性不能直接 如果要使用先设置属性可以访问
    13. id.setAccessible(true);
    14. lastname.setAccessible(true);
    15. gender.setAccessible(true);
    16. firstname.setAccessible(true);
    17. // 给属性赋值
    18. id.set(person,1);
    19. lastname.set(person,"小明");
    20. gender.set(person,"男");
    21. // 静态成员变量 不需要对象就可以赋值
    22. firstname.set(null,"王");
    23. System.out.println(person)

    反射调用对象方法

    1. /*获得一个类的字节码*/
    2. Class pClass = Class.forName("com.bjsxt.demo1.Person");
    3. Constructor cons = pClass.getDeclaredConstructor(int.class, String.class, String.class);
    4. Person person = (Person)cons.newInstance(1, "小明", "男");
    5. // 1执行一个无参数无返回值的方法
    6. // 获取要调用的方法的Method对象
    7. Method showNameMethod = pClass.getDeclaredMethod("showName");
    8. // 让method调用invoke方法 代表让当前方法执行
    9. // 如果是实例方法,在方法执行时,一定需要一个对象才行
    10. // 如果该方法执行需要参数,那么还要传入实参
    11. showNameMethod.invoke(person);
    12. // 2如果执行一个有参数有返回值的方法
    13. // 那么需要在调用时传入实参
    14. Method sumMethod = pClass.getDeclaredMethod("sum", int.class, double.class);
    15. // 设置方法时可以访问的 以免方法是private修饰造成方法不可方法
    16. sumMethod.setAccessible(true);
    17. double res=(double)sumMethod.invoke(person,1,3.14);
    18. System.out.println(res);
    19. // 3执行一个静态方法
    20. // 静态方法可以用对象去调用,也可以用类名去调用
    21. // 所以在执行静态方法是可以不传入对象
    22. Method setFirstname = pClass.getDeclaredMethod("setFirstname", String.class);
    23. setFirstname.invoke(null,"李");
    24. System.out.println(person);

     暴力注入 

    1. color.setAccessible(true);
    2. color.set(dog,"black");
    3. System.out.println(dog.getColor());

      单例模式
      构造器私有化 

    1. // 单例模式
    2. // 1.构造器私有化
    3. Constructor declaredConstructor1 = clazz.getDeclaredConstructor(String.class);
    4. declaredConstructor1.setAccessible(true);
    5. Dog dog1 = declaredConstructor1.newInstance("小强");
    6. System.out.println(dog1.getName());

  • 相关阅读:
    带你认识JDK8中超nice的Native Memory Tracking
    centos8 执行yum install ntpdate命令,报错未找到匹配的参数: ntpdate
    Bert-vits2新版本V2.1英文模型本地训练以及中英文混合推理(mix)
    Xcode 15 编译出错问题解决
    回文串问题
    路由懒加载
    小黑leetcode之旅:95. 至少有 K 个重复字符的最长子串
    897. 最长公共子序列
    2.23C语言学习
    痞子衡嵌入式:IAR环境下无法直接下载调试i.MXRT分散链接工程的解决方案(宏文件.mac+双Flashloader)
  • 原文地址:https://blog.csdn.net/weixin_45777469/article/details/126273469