反射

(可变参数):传参能传0-n个,一个形参只有一个可变参,可变参要放到最后。

类加载器:

    类的加载:

        当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。

        加载:

          就是指将class文件读入内存,并为之创建一个Class对象。任何类被使用时系统都会建立一个Class对象。

        连接:

          验证 是否有正确的内部结构,并和其他类协调一致,准备 负责为类的静态成员分配内存,并设置默认初始化值,解析 将类的二进制数据中的符号引用替换为直接引用。

        初始化:

                  

    类初始化时机:

          1. 创建类的实例

          2. 类的静态变量,或者为静态变量赋值

          3. 类的静态方法

          4. 使用反射方式来强制创建某个类或接口对应的java.lang.Class对象

          5. 初始化某个类的子类

          6. 直接使用java.exe命令来运行某个主类

   

public class Fu {
static{
    System.out.println("父类静态代码块");
}
static int a=1;
public static void a(){
    System.out.println("父类方法");
}
}
public class Zi extends Fu{

}
public class Demo01 {

    public static void main(String[] args) {
        //当调用静态成员,该类进内存
        //System.out.println(Fu.a);
        //调用静态成员方法
        //Fu.a();
        //创建子类对象
        new Zi();
    }

}

    类加载器作用: 负责将.class文件加载到内存中,并为之生成对应的Class对象。

    类加载器的组成:

           Bootstrap ClassLoader 根类加载器,也被称为引导类加载器,负责Java核心类的加载比如System,String等。在JDK中JRE的lib目录下rt.jar文件中。

           Extension ClassLoader 扩展类加载器,负责JRE的扩展目录中jar包的加载。在JDK中JRE的lib目录下ext目录。

           System ClassLoader 系统类加载器(多指我们字节写的类),负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类路径。

反射:

   JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象。

Class类:

    获得Class对象的三种方式:

    

package com.oracle.demo01;

public class Person {
public String name;
private int age;
static{
    System.out.println("静态代码块");
}
public Person(){
    System.out.println("公共空参构造方法");
}
public Person(String name,int age){
    System.out.println("公共有参构造");
}
private Person(int age){
    System.out.println("私有有参构造");
}
public void eat(){
    System.out.println("公共空参方法");
}
public void sleep(String name){
    System.out.println("公共的有参方法");
}
private void play(){
    System.out.println("私有空参方法");
}
@Override
public String toString() {
    return "Person [name=" + name + ", age=" + age + "]";
}

}
public class Demo02 {

    public static void main(String[] args) throws ClassNotFoundException {
        //获取字节码文件对象
        Person p=new Person();
        Class c1=p.getClass();
        System.out.println(c1);
        //通过class属性
        Class c2=Person.class;
        System.out.println(c2);
        //forname方法
        Class c3=Class.forName("com.oracle.demo01.Person");
        System.out.println(c3);
    }

}

  前两种你必须明确Person类型,后面是指定这种类型的字符串就行.这种扩展更强.我不需要知道你的类.我只提供字符串,按照配置文件加载就可以了。

通过反射获取构造方法并使用:

  在反射机制中,把类中的成员(构造方法、成员方法、成员变量)都封装成了对应的类进行表示。其中,构造方法使用类Constructor表示。可通过Class类中提供的方法获取构造方法。  

   返回一个构造方法:

     public Constructor<T> getConstructor(Class<?>... parameterTypes) 获取public修饰, 指定参数类型所对应的构造方法

     public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 获取指定参数类型所对应的构造方法(包含私有的)

   返回多个构造方法

       public Constructor<?>[] getConstructors() 获取所有的public 修饰的构造方法

       public Constructor<?>[] getDeclaredConstructors() 获取所有的构造方法(包含私有的)

通过反射方式,获取构造方法,创建对象:

  步骤:

    1. 获取到Class对象

    2. 获取指定的构造方法

    3. 通过构造方法类Constructor中的方法,创建对象。      public T newInstance(Object... initargs)

通过反射方式,获取私有构造方法,创建对象:

    AccessibleObject 类是 Field、Method 和 Constructor 对象的父类。它提供了将反射的对象标记为在使用时取消默认 Java 语言访问控制检查的能力。对于公共成员、默认(打包)访问成员、受保护成员和私有成员,在分别使用 Field、Method 或 Constructor 对象来设置或获取字段、调用方法,或者创建和初始化类的新实例的时候,会执行访问检查。    

    public void setAccessible(boolean flag) throws SecurityException

    参数值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。参数值为 false 则指示反射的对象应该实施 Java 语言访问检查。

    步骤:

      1. 获取到Class对象

      2. 获取指定的构造方法

      3. 暴力访问, 通过setAccessible(boolean flag)方法

      4. 通过构造方法类Constructor中的方法,创建对象

代码如下:

    

public class Person {
public String name;
private int age;
static{
    System.out.println("静态代码块");
}
public Person(){
    System.out.println("公共空参构造方法");
}
public Person(String name,int age){
    System.out.println("公共有参构造");
}
private Person(int age){
    System.out.println("私有有参构造");
}
public void eat(){
    System.out.println("公共空参构造方法");
}
public void sleep(String name){
    System.out.println("公共的有参方法");
}
private void play(){
    System.out.println("私有空参方法");
}
@Override
public String toString() {
    return "Person [name=" + name + ", age=" + age + "]";
}

}
public class Demo03 {

    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
        //获取Perosn类的字节码文件对象
        //获取公共构造方法
        Class c=Class.forName("com.oracle.demo01.Person");
        //获取公共构造方法
        Constructor con1=c.getConstructor();
        System.out.println(con1);
        //获取公共有参构造
        Constructor con2=c.getDeclaredConstructor(String.class,int.class);
        System.out.println(con2);
        //获取私有有参构造
        Constructor con3=c.getDeclaredConstructor(int.class);
        System.out.println(con3);
        //调用方法创建对象
        Person p1=(Person)con2.newInstance("张三",18);//返回的是Object类型,强转
        p1.eat();
        //取消java语言检查
        //用私有构造方法对象创建Person对象/暴力反射
        //破坏了封装性,不推荐使用
        con3.setAccessible(true);
        Person p2=(Person)con3.newInstance(18);
        p2.eat();
    }

}

通过反射获取成员变量并使用:

      在反射机制中,把类中的成员变量使用类Field表示。可通过Class类中提供的方法获取成员变量。

      返回一个成员变量

      public Field getField(String name) 获取指定的 public修饰的变量

        public Field getDeclaredField(String name) 获取指定的任意变量

      返回多个成员变量

      public Field[] getFields() 获取所有public 修饰的变量

        public Field[] getDeclaredFields() 获取所有的 变量 (包含私有)

通过反射,创建对象,获取指定的成员变量,进行赋值与获取值操作:

      1. 获取Class对象

      2. 获取构造方法

      3. 通过构造方法,创建对象

      4. 获取指定的成员变量(私有成员变量,通过setAccessible(boolean flag)方法暴力访问)

      5. 通过方法,给指定对象的指定成员变量赋值或者获取值

      

public class Person {
public String name;
private int age;
static{
    System.out.println("静态代码块");
}
public Person(){
    System.out.println("公共空参构造方法");
}
public Person(String name,int age){
    System.out.println("公共有参构造");
}
private Person(int age){
    System.out.println("私有有参构造");
}
public void eat(){
    System.out.println("公共空参构造方法");
}
public void sleep(String name){
    System.out.println("公共的有参方法");
}
private void play(){
    System.out.println("私有空参方法");
}
@Override
public String toString() {
    return "Person [name=" + name + ", age=" + age + "]";
}

}
public class Demo04 {

    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException {
        //1:获取字节码文件对象
        Class c=Class.forName("com.oracle.demo01.Person");
        //获取成员变量对象
        Field f=c.getField("name");
        //通过反射创建一个Person对象/快速利用空参构造创建对象方式
        Object obj=c.newInstance();
        //赋值
        f.set(obj,"张三");
        System.out.println(obj);
        System.out.println(f.get(obj));
    }

}

通过反射获取成员方法并使用:

  在反射机制中,把类中的成员方法使用类Method表示。可通过Class类中提供的方法获取成员方法。

  返回获取一个方法:public Method getMethod(String name, Class<?>... parameterTypes),获取public 修饰的方法。

            public Method getDeclaredMethod(String name, Class<?>... parameterTypes), 获取任意的方法,包含私有的

           参数1: name 要查找的方法名称; 参数2: parameterTypes 该方法的参数类型

  返回获取多个方法:

            public Method[] getMethods() 获取本类与父类中所有public 修饰的方法

            public Method[] getDeclaredMethods() 获取本类中所有的方法(包含私有的)

通过反射,创建对象,调用指定的方法:

    获取成员方法,步骤如下:

    1. 获取Class对象

    2. 获取构造方法

    3. 通过构造方法,创建对象

               4. 获取指定的方法

               5. 执行找到的方法

               public Object invoke(Object obj,  Object... args)

    执行指定对象obj中,当前Method对象所代表的方法,方法要传入的参数通过args指定。

    

public class Demo05 {

    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
        //获取字节码文件对象
        Class c=Class.forName("com.oracle.demo01.Person");
        //获取成员方法对象
        Method m=c.getMethod("sleep",String.class);
        //快速创建一个对象
        Object obj=c.newInstance();
        //执行方法
        m.invoke(obj,"张三");
    }

}

反射配置文件:

      

public class Cooker {
public void cook(){
    System.out.println("厨师炒菜");
}
}
public class Worker {
public void work(){
    System.out.println("工人工作");
}
}
public class Student {
public void study(){
    System.out.println("学生学习");
}
}
#ClassName=com.oracle.demo01.Student
#MethodName=study
#ClassName=com.oracle.demo01.Cooker
#MethodName=cook
ClassName=com.oracle.demo01.Worker
MethodName=work
public class Demo06 {

    public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    //创建字符输入流对象,明确数据源
        FileReader fr=new FileReader("src/com/oracle/demo01/pro.properties");
        //创建properties集合
        Properties pro=new Properties();
        //将文件中中的键值对读取到集合中
        pro.load(fr);
        //获取类名
        String className=pro.getProperty("ClassName");
        //获取方法名
        String methodName=pro.getProperty("MethodName");
        //获取字节码文件对象
        Class c=Class.forName(className);
        //创建成员方法对象
        Method m=c.getMethod(methodName);
        //快速创建对象
        Object obj=c.newInstance();
        //执行方法
        m.invoke(obj);
    }

}

 

posted @ 2020-11-07 16:13  马雪峰1  阅读(73)  评论(0编辑  收藏  举报