• 反射和注解


    文章资料来自
    反射的知识
    自定义注解

    反射
    JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制

    与Java反射相关的类

    • Class类:代表类的实体,在运行的Java应用程序中表示类和接口
    • Field类:代表类的成员变量(成员变量也称为类的属性)
    • Method类:代表类的方法
    • Constructor类:代表类的构造方法
    • Annotation类:代表类,方法,变量上的注解

    获取类的三个方法

    //第一种方式获取Class对象  
    Student stu1 = new Student();//这一new 产生一个Student对象,一个Class对象。
    Class stuClass = stu1.getClass();//获取Class对象
    System.out.println(stuClass.getName());
    
    //第二种方式获取Class对象
    Class stuClass2 = Student.class;
    System.out.println(stuClass == stuClass2);//判断第一种方式获取的Class对象和第二种方式获取的是否是同一个
    
    //第三种方式获取Class对象
    try {
        Class stuClass3 = Class.forName("fanshe.Student");//注意此字符串必须是真实路径,就是带包名的类路径,包名.类名
        System.out.println(stuClass3 == stuClass2);//判断三种方式是否获取的是同一个Class对象
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
    

    获取构造方法并使用

    public class Student {
        Student(String str){
            System.out.println("(默认)的构造方法 s = " + str);
        }
        //无参构造方法
        public Student(){
            System.out.println("调用了公有、无参构造方法执行了。。。");
        }
        //有一个参数的构造方法
        public Student(char name){
            System.out.println("姓名:" + name);
        }
        //有多个参数的构造方法
        public Student(String name ,int age){
            System.out.println("姓名:"+name+"年龄:"+ age);//这的执行效率有问题,以后解决。
        }
        //受保护的构造方法
        protected Student(boolean n){
            System.out.println("受保护的构造方法 n = " + n);
        }
        //私有构造方法
        private Student(int age){
            System.out.println("私有的构造方法   年龄:"+ age);
        }	
    }
    
    public static void main(String[] args) {
        //1.加载Class对象
        Class clazz = Class.forName("com.Student");
        //2.获取所有公有构造方法
        System.out.println("****所有公有构造方法******");
        Constructor[] conArray = clazz.getConstructors();
        for(Constructor c : conArray){
            System.out.println(c);
        }
        System.out.println("****所有的构造方法(包括:私有、受保护、默认、公有)****");
        conArray = clazz.getDeclaredConstructors();
        for(Constructor c : conArray){
            System.out.println(c);
        }
        System.out.println("*******获取公有、无参的构造方法*******");
        Constructor con = clazz.getConstructor(null);
        //1>、因为是无参的构造方法所以类型是一个null,不写也可以:这里需要的是一个参数的类型,切记是类型
        //2>、返回的是描述这个无参构造函数的类对象。
        System.out.println("con = " + con);
        //调用构造方法
        Object obj = con.newInstance();
        //	System.out.println("obj = " + obj);
        //	Student stu = (Student)obj;
        System.out.println("****获取私有构造方法,并调用****");
        con = clazz.getDeclaredConstructor(char.class);
        System.out.println(con);
        //调用构造方法
        con.setAccessible(true);//暴力访问(忽略掉访问修饰符)
        obj = con.newInstance('男');
    }
    

    成员变量并调用

    public class Student {
        public String name;
        protected int age;
        char sex;
        private String phoneNum;
    
        @Override
        public String toString() {
            return "Student [name=" + name + ", age=" + age + ", sex=" + sex
                    + ", phoneNum=" + phoneNum + "]";
        }	
    }
    
    public static void main(String[] args) {
        //3.获取字段
        System.out.println("****获取所有公有的字段******");
        Field[] fieldArray = clazz.getFields();
        for(Field f : fieldArray){
            System.out.println(f);
        }
        System.out.println("************获取所有的字段(包括私有、受保护、默认的)********************");
        fieldArray = clazz.getDeclaredFields();
        for(Field f : fieldArray){
            System.out.println(f);
        }
        System.out.println("***获取公有字段**并调用*****");
        Field f = clazz.getField("name");
        System.out.println(f);
        //获取一个对象
        Object obj = clazz.getConstructor().newInstance();//产生Student对象--》Student stu = new Student();
        //为字段设置值
        f.set(obj, "刘德华");//为Student对象中的name属性赋值--》stu.name = "刘德华"
        //验证
        Student stu = (Student)obj;
        System.out.println("验证姓名:" + stu.name);
    
        System.out.println("********获取私有字段****并调用******");
        f = clazz.getDeclaredField("phoneNum");
        System.out.println(f);
        f.setAccessible(true);//暴力反射,解除私有限定
        f.set(obj2, "18888889999");
        System.out.println("验证电话:" + stu);
    }
    

    成员方法并调用

    public class Student {
            //1.获取Class对象
        Class stuClass = Class.forName("fanshe.method.Student");
        //2.获取所有公有方法
        System.out.println("****获取所有的公有方法********");
        stuClass.getMethods();
        Method[] methodArray = stuClass.getMethods();
        for(Method m : methodArray){
            System.out.println(m);
        }
        System.out.println("******获取所有的方法,包括私有的*******");
        methodArray = stuClass.getDeclaredMethods();
        for(Method m : methodArray){
            System.out.println(m);
        }
        System.out.println("*******获取公有的show1()方法********");
        Method m = stuClass.getMethod("show1", String.class);
        System.out.println(m);
        //实例化一个Student对象
        Object obj = stuClass.getConstructor().newInstance();
        m.invoke(obj, "刘德华");
    
        System.out.println("*******获取私有的show4()方法*********");
        m = stuClass.getDeclaredMethod("show4", int.class);
        System.out.println(m);
        m.setAccessible(true);//解除私有限定
        Object result = m.invoke(obj, 20);//需要两个参数,一个是要调用的对象(获取有反射),一个是实参
        System.out.println("返回值:" + result);
    }
    

    自定义注解
    你可以理解为一个标记,通过自定义一个标记,把标记写到类上,方法上,属性上后,通过反射判断是否带有这个标记,再赋予标记的存在意义

    // 自定义注解
    @Target(ElementType.FIELD)  //  注解用于字段上
    @Retention(RetentionPolicy.RUNTIME)  // 保留到运行时,可通过注解获取
    public @interface MyField {
        String description();
        int length();
    }
    
    public class MyFieldTest {
        //使用我们的自定义注解
        @MyField(description = "用户名", length = 12)
        private String username;
    
        public static void main(String[] args){
            // 获取类模板
            Class c = MyFieldTest.class;
    
            // 获取所有字段
            for(Field f : c.getDeclaredFields()){
                // 判断这个字段是否有MyField注解
                if(f.isAnnotationPresent(MyField.class)){
                    MyField annotation = f.getAnnotation(MyField.class);
                    System.out.println("字段:[" + f.getName() + "], 描述:[" + annotation.description() + "], 长度:[" + annotation.length() +"]");
                }
            }
        }
    }
    
  • 相关阅读:
    php基本语法与安装
    面向对象编程 es5和es6的构造函数
    利用正则搜索替换
    正则特殊符号
    正则边界符 限定符
    面试官给我挖坑:rm删除文件之后,空间就被释放了吗?
    为什么 IPv6 难以取代 IPv4
    Docker系列教程04-Docker构建镜像的三种方式
    Docker系列教程03-Docker私有仓库搭建(registry)
    Linux-I/O模型详解
  • 原文地址:https://www.cnblogs.com/pengdt/p/12240532.html
Copyright © 2020-2023  润新知