Java基础加强01-反射2

1、反射的概念

  反射是框架设计的灵魂。

  框架:半成品软件。可以在框架的基础上进行软件开发,简化编码。

  使用框架并不需要使用反射。但是写框架的时候是需要反射的。

  将类的各个组成部分封装为其他对象,这就是反射。

2、Java代码在计算机中经历的阶段:三个阶段

 第一阶段:Source源代码阶段

  Person.java

      public class Person{

        private String name;

        private int age;

        public Person(){}

        public void eat(){}

      }

   编译:javac 编译,编译完生成Person.class字节码文件

   Person.class中包含三部分:

        a. private String name;

          private int age;

        b. public Person(){}

        c. public void eat(){}

  第二阶段:Class类对象阶段。

       类加载器ClassLoader,可以把Person.class加载到内存中,加载到内存后有对应的Class类对象

          成员变量,封装为Field对象。会有多个成员变量,用Field[] fields;

          构造方法,封装为Constructor对象。可能有多个构造方法,用Constructor[] cons;

          成员方法,封装为Method对象。有多个成员方法,用Method[] methods。

       将来可以通过Class类对象的行为来创建真正的Person等对象。

  第三阶段:Runtime运行时阶段。

          Person对象

       new Person();

  

3、反射的好处

   a. 可以在程序运行过程中,操作这些对象。

   b. 可以解耦,提高程序的可扩展性。

4、获取Class类对象的方式

  • 将字节码文件加载进内存,返回Class对象。

    多用于配置文件,将类名定义在配置文件中。读取文件,加载类。

    Java代码在第一个阶段,也就是只有.class字节码文件并没有进内存。需要手动将.class加载进内存,通过

    Class.forName("全类名");

  • 已经加载进内存了,可通过类名的属性class获取Class对象。

    多用于参数的传递。

    类名.class

  • 已经有对象了,通过getClass()方法(在Object类中定义着)获取Class对象。

    多用于对象的获取字节码的方式。

    对象.getClass()

package com.itheima.day01.reflect;

public class Person {
    private  String name;
    private int age;

    //快速创建构造方法及getter\setter\toString(),Alt+Insert
    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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
package com.itheima.day01.reflect;

public class ReflectDemo1 {
    /**
     获取Class对象的方式:
     1. Class.forName("全类名"):将字节码文件加载进内存,返回Class对象
     2. 类名.class:通过类名的属性class获取
     3. 对象.getClass():getClass()方法在Object类中定义着。
     */
    public static void main(String[] args) throws ClassNotFoundException {
        //Person.java保存后自动编译为Person.class
        //1. Class.forName("全类名")
        Class cls1 = Class.forName("com.itheima.day01.reflect.Person");
        System.out.println(cls1);//class com.itheima.day01.reflect.Person

        // 2. 类名.class
        Class cls2 = Person.class;
        System.out.println(cls2);//class com.itheima.day01.reflect.Person

        //3. 对象.getClass()
        Person p = new Person();
        Class cls3 = p.getClass();
        System.out.println(cls3);//class com.itheima.day01.reflect.Person

        //==比较三个对象
        System.out.println(cls1== cls2);//true
        System.out.println(cls1==cls3);//true
    }
}

结论:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。

5、Class对象功能

  • 获取成员变量们

    Field[] getFields():获取所有公共的成员变量

    Field getField(String name):获取指定名称的公共成员变量

    Field[] getDeclaredFields():获取所有的成员变量

    Field getDeclaredField(String name):获取单个成员变量

  • 获取构造方法们

    Constructor<?>[] getConstructors():获取所有公共的构造方法

    Constructor<T> getConstructor(类<?>... parameterTypes):获取指定的公共构造方法

    Constructor<?>[] getDeclaredConstructors():获取所有的构造方法

    Constructor<T> getDeclaredConstructor(类<?>... parameterTypes):获取单个构造方法

  • 获取成员方法们

    Method[] getMethods():获取所有的公共成员方法,包括Object的方法

    Method getMethod(String name, 类<?>... parameterTypes):获取指定名称的公共成员方法

    Method[] getDeclaredMethods():获取所有的成员方法

    Method getDeclaredMethod(String name, 类<?>... parameterTypes):获取单个成员方法

  • 获取类名

    String getName():获取类名

    String getPackage():获取此类的包

6、使用Class对象功能

  • Field:成员变量

    操作:

       1、设置值:void set(Object obj, Object value)

     2、获取值:Object get(Object obj)

     3、忽略访问权限修饰符的安全检查:setAccessible(true)暴力反射

package com.itheima.day01.reflect;

public class Person {
    private  String name;
    private int age;
    public String a;
    protected String b;
    String c;
    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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", a='" + a + '\'' +
                ", b='" + b + '\'' +
                ", c='" + c + '\'' +
                ", d='" + d + '\'' +
                '}';
    }
}
package com.itheima.day01.reflect;

import java.lang.reflect.Field;

public class ReflectDemo2 {
 
    public static void main(String[] args) throws Exception {
        //0.获取Person的Class对象
        Class personClass = Person.class;

        Field[] fields= personClass.getFields();
        for(Field field: fields){
            //public java.lang.String com.itheima.day01.reflect.Person.a
            System.out.println(field);
        }

        System.out.println("-------------");

        Field a= personClass.getField("a");
        //获取成员变量a的值
        Person p = new Person();
        Object value = a.get(p);
        System.out.println(value);//null

        a.set(p,"abc");
        System.out.println(p);//Person{name='null', age=0, a='abc', b='null', c='null', d='null'}

        System.out.println("==================");

        Field[] declaredFields = personClass.getDeclaredFields();
        for(Field field: declaredFields){
//            private java.lang.String com.itheima.day01.reflect.Person.name
//            private int com.itheima.day01.reflect.Person.age
//            public java.lang.String com.itheima.day01.reflect.Person.a
//            protected java.lang.String com.itheima.day01.reflect.Person.b
//            java.lang.String com.itheima.day01.reflect.Person.c
//            private java.lang.String com.itheima.day01.reflect.Person.d
            System.out.println(field);
        }

        Field d = personClass.getDeclaredField("d");
        //忽略访问权限修饰符的安全检查
        d.setAccessible(true);//暴力反射
        Object value2 = d.get(p);
        System.out.println(value2);//null
    }
}
  • Constructor:构造方法

    构造方法用来创建对象

      1、创建对象:T newInstance(Object... initargs)

      2、如果使用空参数构造方法创建对象,操作可以简化:Class对象的newInstance()方法      

      3、忽略访问权限修饰符的安全检查:setAccessible(true)暴力反射

package com.itheima.day01.reflect;

import java.lang.reflect.Constructor;

public class ReflectDemo3 {
    public static void main(String[] args) throws Exception {
        //0.获取Person的Class对象
        Class personClass = Person.class;

        Constructor[] constructors = personClass.getConstructors();
        for(Constructor constructor:constructors){
//            public com.itheima.day01.reflect.Person()
//            public com.itheima.day01.reflect.Person(java.lang.String,int)
            System.out.println(constructor);
        }

        Constructor constructor = personClass.getConstructor(String.class,int.class);
        //public com.itheima.day01.reflect.Person(java.lang.String,int)
        System.out.println(constructor);

        //创建对象
        Object person = constructor.newInstance("ab",23);
        //Person{name='ab', age=23, a='null', b='null', c='null', d='null'}
        System.out.println(person);

        System.out.println("-----------------");

        Constructor constructor1 = personClass.getConstructor();
        //public com.itheima.day01.reflect.Person()
        System.out.println(constructor1);

        //创建对象
        Object person1 = constructor1.newInstance();
        //Person{name='null', age=0, a='null', b='null', c='null', d='null'}
        System.out.println(person1);

        //如果使用空参数构造方法创建对象,操作可以简化:Class对象的newInstance方法
        Object person2 =  personClass.newInstance();
        //Person{name='null', age=0, a='null', b='null', c='null', d='null'}
        System.out.println(person2);

        constructor.setAccessible(true);
    }
}
  • Method:成员方法

    操作:    

      1、执行方法:Object invoke(Object obj, Object... args)

      2、忽略访问权限修饰符的安全检查:setAccessible(true)暴力反射

      3、获取方法名称:String getName()

      4、获取该方法的参数类型:Class getParameterTypes()

package com.itheima.day01.reflect;

public class Person {
    private  String name;
    private int age;
    public String a;
    protected String b;
    String c;
    private String d;

    //快速创建构造方法及getter\setter\toString(),Alt+Insert
    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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", a='" + a + '\'' +
                ", b='" + b + '\'' +
                ", c='" + c + '\'' +
                ", d='" + d + '\'' +
                '}';
    }

    public void eat(){
        System.out.println("eat...");
    }

    public void eat(String food){
        System.out.println("eat..."+ food);
    }
}
package com.itheima.day01.reflect;

import java.lang.reflect.Method;

public class ReflectDemo4 {
    public static void main(String[] args) throws Exception {
        Class personClass = Person.class;
        
        //获取指定名称的成员方法
        Method eatMethod =personClass.getMethod("eat");

        Person p = new Person();
        //执行方法
        eatMethod.invoke(p);//输出:eat...

        Method eatMethod1 =personClass.getMethod("eat",String.class);
        eatMethod1.invoke(p,"chifan"); //输出:eat...chifan
     Class<?>[] parameterTypes = eatMethod1.getParameterTypes();
     for(Class parameterType : parameterTypes){
      System.out.println(parameterType);//class java.lang.String
      System.out.println(parameterType.getName());//java.lang.String
     }  
        System.out.println("------------");

        //获取所有public修饰的方法
        Method[] methods = personClass.getMethods();
        for(Method method:methods){
//            public java.lang.String com.itheima.day01.reflect.Person.toString()
//            public java.lang.String com.itheima.day01.reflect.Person.getName()
//            public void com.itheima.day01.reflect.Person.setName(java.lang.String)
//            public void com.itheima.day01.reflect.Person.eat()
//            public void com.itheima.day01.reflect.Person.eat(java.lang.String)
//            public int com.itheima.day01.reflect.Person.getAge()
//            public void com.itheima.day01.reflect.Person.setAge(int)
//            public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
//            public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
//            public final void java.lang.Object.wait() throws java.lang.InterruptedException
//            public boolean java.lang.Object.equals(java.lang.Object)
//            public native int java.lang.Object.hashCode()
//            public final native java.lang.Class java.lang.Object.getClass()
//            public final native void java.lang.Object.notify()
//            public final native void java.lang.Object.notifyAll()
            System.out.println(method);
            String name = method.getName();
//            toString
//            getName
//            setName
//            getAge
//            setAge
//            eat
//            eat
//            wait
//            wait
//            wait
//            equals
//            hashCode
//            getClass
//            notify
//            notifyAll
            System.out.println(name);
            method.setAccessible(true);
        }

        //获取类名
        String className = personClass.getName();
        System.out.println(className);                //输出:com.itheima.day01.reflect.Person
        System.out.println(personClass.getPackage()); //输出:package com.itheima.day01.reflect
    }
}

7、案例

   写一个“框架”,不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法

  实现:

    a. 配置文件

    b. 反射

  步骤:

    a. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中

    b.在程序中加载读取配置文件

    c. 使用反射技术来加载文件进内存

    d. 创建对象

    e. 执行方法

 src下创建一个pro.properties配置文件

package com.itheima.day01.reflect;

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.加载配置文件
        //1.1 创建Properties对象
        Properties pro = new Properties();
        //1.2 加载配置文件,转换为一个集合(Properties是一个Map集合类)
        //1.3 获取class目录下的配置文件
        ClassLoader classLoader= ReflectTest.class.getClassLoader();//获取ReflectTest.class文件的类加载器
     System.out.println(classLoader);  //jdk.internal.loader.ClassLoaders$AppClassLoader@2437c6dc InputStream is
= classLoader.getResourceAsStream("pro.properties");//从classpath的根路径下查找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 method = cls.getMethod(methodName); //6. 执行方法 method.invoke(obj);//Person类的eat方法被执行,输出:eat... //不改变此类的任何方法,只需要改pro.properties配置文件即可 //系统庞大,改Java代码需要重新测试编译上线,改配置文件改完就可以了,扩展性更强 } }

 8、总结与补充

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

         三种方式常用第三种,第一种对象都有了还要反射干什么。第二种需要导入类的包,依赖太强,不导包就抛编译错误。一般都第三种,一个字符串可以传入也可写在配置文件中等多种方法。

  • 构造器可以提供许多特殊的方法,构造器作为一种方法,负责类中成员变量(域)的初始化。实例构造器分为缺省构造器和非缺省构造器。

    构造器最大的用处就是在创建对象时执行初始化,当创建一个对象时,系统会为这个对象的实例进行默认的初始化。如果想改变这种默认的初始化,就可以通过自定义构造器来实现。

posted on 2021-09-18 00:15  花溪月影  阅读(58)  评论(0编辑  收藏  举报