JAVA-反射

反射

  1. 反射_获取字节码Class对象的三种方式

    • ReflectDemo1

      package cn.xiaoge.day22.reflect;
      
      /*
          反射: 框架设计的灵魂
              框架: 半成品软件. 可以在框架的基础上进行软件开发, 简化编码
              反射: 将类的各个组成部分封装为其他对象, 这就是反射机制
                  好处:
                      1. 可以在程序运行过程中, 操作这些对象
                      2. 可以解耦, 提高程序的可扩展性
              获取class对象的方式:
                  1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象
                      - 多用于配置文件, 将类名定义在配置文件中. 读取文件, 加载类
      
                  2. 类名.class: 通过类名的属性class获取
                      - 多用于参数的传递
      
                  3. 对象.getClass(): getClass()方法在Object类中定义着.
                      - 多用于对象的获取字节码的方式
      
                  结论:
                      同一个字节码文件(*.class)在一次程序运行过程中, 只会被加载一次, 不论通过哪一种方式获取的Class对象都是同一个.
      
      
          Class对象功能:  不带Declared只能获取公共的就是public的, 带Declared就是获取所有的
              获取功能:
                  1. 获取成员变量们
                      - Field[] getFields(): 获取所有public修饰的成员变量的
                      - Field getField(String name): 获取指定名称的 public修饰的成员变量的
      
                      - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
                      - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符
                  2. 获取构造方法们
                      - Constructor<?>[] getConstructors()
                      - Constructor<T> getConstructor(类<?>... parameterTypes)
      
                      - Constructor<?>[] getDeclaredConstructors()
                      - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
                  3. 获取成员方法们
                      - Method[] getMethods()
                      - Method getMethod(String name, 类<?>... parameterTypes)
      
                      - Method[] getDeclaredMethods()
                      - Method getDeclaredMethod(String name, 类<?>... parameterTypes)
                  4. 获取类名
                      - String getName()
      
              Field: 成员变量
                  操作:
                      1. 设置值
                          - void set(Object obj, Object value)
                      2. 获取值
                          - get(Object obj)
                      3. 忽略访问权限修饰符的安全检查
                          - d.setAccessible(true): 暴力反射
              Constructor: 构造方法
                  创建对象:
                      - T newInstance(Object... initargs)
      
                      - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法
      
              Method: 方法对象
                  执行方法:
                      - Object invoke(Object obj, Object... args)
      
                  获取方法名称:
                      - String getName(): 获取方法名
      
              获取类名:
                  Class对象.getName();
       */
      
      import cn.xiaoge.day22.domain.Person;
      import cn.xiaoge.day22.domain.Student;
      
      public class ReflectDemo1 {
      
          /**
           * 获取class对象的方式:
           *     1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象
           *     2. 类名.class: 通过类名的属性class获取
           *     3. 对象.getClass(): getClass()方法在Object类中定义着.
           */
          public static void main(String[] args) throws Exception {
      
              // 1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象
              Class cls1 = Class.forName("cn.xiaoge.day22.domain.Person");
              System.out.println(cls1);
      
              // 2. 类名.class: 通过类名的属性class获取
              Class cls2 = Person.class;
              System.out.println(cls2);
      
              // 3. 对象.getClass(): getClass()方法在Object类中定义着.
              Class cls3 = new Person().getClass();
              System.out.println(cls3);
      
              // == 比较三个对象
              System.out.println(cls1 == cls2); // true
              System.out.println(cls1 == cls3); // true
      
              Class c = Student.class;
              System.out.println(c == cls1); // false
          }
      
      
      }
      
      
      // 运行结果
      class cn.xiaoge.day22.domain.Person
      class cn.xiaoge.day22.domain.Person
      class cn.xiaoge.day22.domain.Person
      true
      true
      false
      
  2. 反射_Class对象功能获取Field

    • ReflectDemo2

      package cn.xiaoge.day22.reflect;
      
      import cn.xiaoge.day22.domain.Person;
      
      import java.lang.reflect.Field;
      
      public class ReflectDemo2 {
          private Class personClass;
      
          /*
              Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的
              获取功能:
                  1. 获取成员变量们
                      - Field[] getFields(): 获取所有public修饰的成员变量的
                      - Field getField(String name): 获取指定名称的 public修饰的成员变量的
      
                      - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
                      - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符
                  2. 获取构造方法们
                      - Constructor<?>[] getConstructors()
                      - Constructor<T> getConstructor(类<?>... parameterTypes)
      
                      - Constructor<?>[] getDeclaredConstructors()
                      - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
                  3. 获取成员方法们
                      - Method[] getMethods()
                      - Method getMethod(String name, 类<?>... parameterTypes)
      
                      - Method[] getDeclaredMethods()
                      - Method getDeclaredMethod(String name, 类<?>... parameterTypes)
                  4. 获取类名
                      - String getName()
      
              Field: 成员变量
                  操作:
                      1. 设置值
                          - void set(Object obj, Object value)
                      2. 获取值
                          - get(Object obj)
                      3. 忽略访问权限修饰符的安全检查
                          - d.setAccessible(true): 暴力反射
      
              Constructor: 构造方法
                  创建对象:
                      - T newInstance(Object... initargs)
      
                      - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法
      
              Method: 方法对象
                  执行方法:
                      - Object invoke(Object obj, Object... args)
      
                  获取方法名称:
                      - String getName(): 获取方法名
      
              获取类名:
                  Class对象.getName();
           */
      
          public static void main(String[] args) throws Exception {
              // 1. 获取Person的Class对象
              Class personClass = Person.class;
      
              /*
                  1. 获取成员变量们
                      - Field[] getFields() : 获取所有public修饰的成员变量的
                      - Field getField(String name): 获取指定名称的 public修饰的成员变量的
      
                      - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
                      - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符
                  Field: 成员变量
                  操作:
                      1. 设置值
                          - void set(Object obj, Object value)
                      2. 获取值
                          - get(Object obj)
                      3. 忽略访问权限修饰符的安全检查
                          - d.setAccessible(true): 暴力反射
               */
              // 1. Field[] getFields() : 获取所有public修饰的成员变量的
              Field[] fields = personClass.getFields();
              for (Field field : fields) {
                  System.out.println(field);
              }
      
              System.out.println("----------------------");
              // 2. Field getField(String name): 获取指定名称的 public修饰的成员变量的
              Field a = personClass.getField("a");
      
              // 获取成员变量a的值
              Person p = new Person();
              Object value = a.get(p);
              System.out.println(value);
      
              // 设置成员变量a的值
              a.set(p, "10");
              System.out.println(p);
      
              System.out.println("===================================");
      
              // 3. Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
              Field[] declaredFields = personClass.getDeclaredFields();
              for (Field declaredField : declaredFields) {
                  System.out.println(declaredField);
              }
      
              // 4. Field getDeclaredField(String name)
              Field d = personClass.getDeclaredField("d");
              // 忽略访问权限修饰符的安全检查
              d.setAccessible(true); // 暴力反射
              Object value2 = d.get(p);
              System.out.println(value2);
          }
      
      }
      
      
      // 运行结果
      public java.lang.String cn.xiaoge.day22.domain.Person.a
      ----------------------
      null
      Person{name='null', age=0, a='10', b='null', c='null', d='null'}
      ===================================
      private java.lang.String cn.xiaoge.day22.domain.Person.name
      private int cn.xiaoge.day22.domain.Person.age
      public java.lang.String cn.xiaoge.day22.domain.Person.a
      protected java.lang.String cn.xiaoge.day22.domain.Person.b
      java.lang.String cn.xiaoge.day22.domain.Person.c
      private java.lang.String cn.xiaoge.day22.domain.Person.d
      null
      
  3. 反射_Class对象功能获取Constructor

    • ReflectDemo3

      package cn.xiaoge.day22.reflect;
      
      import cn.xiaoge.day22.domain.Person;
      
      import java.lang.reflect.Constructor;
      
      public class ReflectDemo3 {
          private Class personClass;
      
          /*
              Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的
              获取功能:
                  1. 获取成员变量们
                      - Field[] getFields(): 获取所有public修饰的成员变量的
                      - Field getField(String name): 获取指定名称的 public修饰的成员变量的
      
                      - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
                      - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符
                  2. 获取构造方法们
                      - Constructor<?>[] getConstructors()
                      - Constructor<T> getConstructor(类<?>... parameterTypes)
      
                      - Constructor<?>[] getDeclaredConstructors()
                      - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
                  3. 获取成员方法们
                      - Method[] getMethods()
                      - Method getMethod(String name, 类<?>... parameterTypes)
      
                      - Method[] getDeclaredMethods()
                      - Method getDeclaredMethod(String name, 类<?>... parameterTypes)
                  4. 获取类名
                      - String getName()
      
              Field: 成员变量
                  操作:
                      1. 设置值
                          - void set(Object obj, Object value)
                      2. 获取值
                          - get(Object obj)
                      3. 忽略访问权限修饰符的安全检查
                          - d.setAccessible(true): 暴力反射
      
              Constructor: 构造方法
                  创建对象:
                      - T newInstance(Object... initargs)
      
                      - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法
      
              Method: 方法对象
                  执行方法:
                      - Object invoke(Object obj, Object... args)
      
                  获取方法名称:
                      - String getName(): 获取方法名
      
              获取类名:
                  Class对象.getName();
           */
      
          public static void main(String[] args) throws Exception {
              // 1. 获取Person的Class对象
              Class personClass = Person.class;
      
              /*
                 2. 获取构造方法们
                      - Constructor<?>[] getConstructors()
                      - Constructor<T> getConstructor(类<?>... parameterTypes)
      
                      - Constructor<?>[] getDeclaredConstructors()
                      - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
               */
      
      
              // Constructor<T> getConstructor(类<?>... parameterTypes)
              Constructor c = personClass.getConstructor(String.class, int.class);
              System.out.println(c);
              // 创建对象
              Object person = c.newInstance("张三", 23);
              System.out.println(person);
      
      
              System.out.println("=====================================");
      
      
              // Constructor<T> getConstructor(类<?>... parameterTypes)
              Constructor c1 = personClass.getConstructor();
              System.out.println(c);
              // 创建对象
              Object person1 = c1.newInstance();
              System.out.println(person1);
      
      
              System.out.println("=====================================");
      
      
              // 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法
              Object person2 = personClass.newInstance();
              System.out.println(person2);
      
      
              System.out.println("=====================================");
      
      
              Constructor[] cs = personClass.getConstructors();
              for (Constructor constructor : cs) {
                  System.out.println(constructor);
              }
              Object person3 = cs[0].newInstance();
              System.out.println(person3);
              Object person4 = cs[1].newInstance("李四", 33);
              System.out.println(person4);
          }
      
      }
      
      
      // 运行结果
      public cn.xiaoge.day22.domain.Person(java.lang.String,int)
      Person{name='张三', age=23, a='null', b='null', c='null', d='null'}
      =====================================
      public cn.xiaoge.day22.domain.Person(java.lang.String,int)
      Person{name='null', age=0, a='null', b='null', c='null', d='null'}
      =====================================
      Person{name='null', age=0, a='null', b='null', c='null', d='null'}
      =====================================
      public cn.xiaoge.day22.domain.Person()
      public cn.xiaoge.day22.domain.Person(java.lang.String,int)
      Person{name='null', age=0, a='null', b='null', c='null', d='null'}
      Person{name='李四', age=33, a='null', b='null', c='null', d='null'}
      
  4. 反射_Class对象功能获取Method

    • ReflectDemo4

      package cn.xiaoge.day22.reflect;
      
      import cn.xiaoge.day22.domain.Person;
      
      import java.lang.reflect.Constructor;
      import java.lang.reflect.Method;
      
      public class ReflectDemo4 {
          private Class personClass;
      
          /*
              Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的
              获取功能:
                  1. 获取成员变量们
                      - Field[] getFields(): 获取所有public修饰的成员变量的
                      - Field getField(String name): 获取指定名称的 public修饰的成员变量的
      
                      - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符
                      - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符
                  2. 获取构造方法们
                      - Constructor<?>[] getConstructors()
                      - Constructor<T> getConstructor(类<?>... parameterTypes)
      
                      - Constructor<?>[] getDeclaredConstructors()
                      - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
                  3. 获取成员方法们
                      - Method[] getMethods()
                      - Method getMethod(String name, 类<?>... parameterTypes)
      
                      - Method[] getDeclaredMethods()
                      - Method getDeclaredMethod(String name, 类<?>... parameterTypes)
                  4. 获取类名
                      - String getName()
      
              Field: 成员变量
                  操作:
                      1. 设置值
                          - void set(Object obj, Object value)
                      2. 获取值
                          - get(Object obj)
                      3. 忽略访问权限修饰符的安全检查
                          - d.setAccessible(true): 暴力反射
      
              Constructor: 构造方法
                  创建对象:
                      - T newInstance(Object... initargs)
      
                      - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法
      
              Method: 方法对象
                  执行方法:
                      - Object invoke(Object obj, Object... args)
      
                  获取方法名称:
                      - String getName(): 获取方法名
      
              获取类名:
                  Class对象.getName();
           */
      
      
          public static void main(String[] args) throws Exception {
              // 1. 获取Person的Class对象
              Class personClass = Person.class;
      
              /*
                 3. 获取成员方法们
                      - Method[] getMethods()
                      - Method getMethod(String name, 类<?>... parameterTypes)
      
                      - Method[] getDeclaredMethods()
                      - Method getDeclaredMethod(String name, 类<?>... parameterTypes)
               */
              // 获取指定名称的方法
              Method eat_method = personClass.getMethod("eat");
              Person p = new Person();
              // 执行方法
              eat_method.invoke(p);
      
              System.out.println("=========================");
      
              Method eat_method2 = personClass.getMethod("eat", String.class);
              // 执行方法
              eat_method2.invoke(p, "鸡");
      
      
              System.out.println("=========================");
      
              // 获取所有public修饰的方法
              Method[] methods = personClass.getMethods();
              for (Method method : methods) {
                  System.out.println(method);
                  System.out.println(method.getName());
              }
      
      
              System.out.println("=========================");
              // 获取类名
              String className = personClass.getName();
              System.out.println(className);
          }
      
      }
      
      
      // 运行结果
      eat....
      =========================
      eat:=========================
      public java.lang.String cn.xiaoge.day22.domain.Person.toString()
      toString
      public java.lang.String cn.xiaoge.day22.domain.Person.getName()
      getName
      public void cn.xiaoge.day22.domain.Person.setName(java.lang.String)
      setName
      public int cn.xiaoge.day22.domain.Person.getAge()
      getAge
      public void cn.xiaoge.day22.domain.Person.setAge(int)
      setAge
      public void cn.xiaoge.day22.domain.Person.eat(java.lang.String)
      eat
      public void cn.xiaoge.day22.domain.Person.eat()
      eat
      public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
      wait
      public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
      wait
      public final void java.lang.Object.wait() throws java.lang.InterruptedException
      wait
      public boolean java.lang.Object.equals(java.lang.Object)
      equals
      public native int java.lang.Object.hashCode()
      hashCode
      public final native java.lang.Class java.lang.Object.getClass()
      getClass
      public final native void java.lang.Object.notify()
      notify
      public final native void java.lang.Object.notifyAll()
      notifyAll
      =========================
      cn.xiaoge.day22.domain.Person
      
  5. 案例

    • 配置文件pro.properties

      className=cn.xiaoge.day22.domain.Student
      methodName=sleep
      
    • domain包下的两个类

      • Person
        package cn.xiaoge.day22.domain;
        
        public class Person {
        
            private String name;
            private int age;
            public String a;
            protected String b;
            String c; // 默认default
            private String d;
        
            public Person() {
            }
        
            public Person(String name, int age) {
                this.name = name;
                this.age = age;
            }
        
            public String getName() {
                return name;
            }
        
            public void setName(String name) {
                this.name = name;
            }
        
            public int getAge() {
                return age;
            }
        
            public void setAge(int age) {
                this.age = age;
            }
        
            public void eat(){
                System.out.println("eat....");
            }
        
            public void eat(String food){
                System.out.println("eat:" + food);
            }
        
            @Override
            public String toString() {
                return "Person{" +
                        "name='" + name + '\'' +
                        ", age=" + age +
                        ", a='" + a + '\'' +
                        ", b='" + b + '\'' +
                        ", c='" + c + '\'' +
                        ", d='" + d + '\'' +
                        '}';
            }
        }
        
        
      • Student
        package cn.xiaoge.day22.domain;
        
        public class Student {
        
            public void sleep(){
                System.out.println("sleep...");
            }
        
        }
        
        
    • ReflectTest

      package cn.xiaoge.day22.reflect;
      
      /*
          需求: 写一个"框架", 不能改变该类的任何代码, 可以帮我们创建任意类的独享, 并且执行其中任意方法
              实现:
                  1. 配置文件
                  2. 反射
              步骤:
                  1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
                  2. 在程序中加载读取配置文件
                  3. 使用反射技术来加载类文件进内存
                  4. 创建对象
                  5. 执行方法
       */
      
      import java.io.InputStream;
      import java.lang.reflect.Method;
      import java.util.Properties;
      
      public class ReflectTest {
          /**
           *  框架类
           */
          public static void main(String[] args) throws Exception {
              // 可以创建任意类的对象, 可以执行任意方法
      
              /*
                  前提: 不能改变该类的任何代码, 可以创建任意类的对象, 可以执行任意方法
               */
      
              // 1. 加载配置文件, 使用Properties里面的load方法把pro.properties加载到内存中, 返回一个集合
              // 1.1 创建Properties对象
              Properties pro = new Properties();
              // 1.2 加载配置文件, 转换为一个集合
              // 1.2.1 获取class目录下的配置文件
              ClassLoader classLoader = ReflectTest.class.getClassLoader();// 获取这个字节码文件对应的类加载器
              InputStream is = classLoader.getResourceAsStream("pro.properties");// 返回获取资源对应的字节流
              pro.load(is);
      
              // 2. 获取配置文件中定义的数据  获取对应文件里面的键所对应的值
              String className = pro.getProperty("className");
              String methodName = pro.getProperty("methodName");
      
              // 3. 加载该类文件进内存
              Class cls = Class.forName(className);
      
              // 4. 创建对象
              Object obj = cls.newInstance();
      
              // 5. 获取方法对象
              Method methodObj = cls.getMethod(methodName);
      
              // 6. 执行方法
              methodObj.invoke(obj);
      
          }
      
      }
      
      
      // 运行结果
      sleep...
      
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

只因为你温柔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值