反射

  1. 反射

Java代码和Java文件

Java代码根本格局    1. Java代码都在类内或者接口内    2.         class 类名 {            成员变量            构造方法            成员办法            Annotation 注解        }Java文件要求:    1. 通常状况下一个Java文件对应一个Java类    2. Java文件蕴含以后Java代码的所有内容!!!

Java文件和.class字节码文件

Java文件    FirstJava.java    通过编译器 javac ==> javac FirstJava.java ==> FirstJava.class.class字节码文件是什么???    二进制可执行文件。    .class字节码文件中会蕴含Java文件的所有内容。    .class字节码文件蕴含Java程序的所有可执行内容(正文不参加编译和执行)。

class字节码文件在内存中的地位

class字节码文件和Java代码关系

Class类相干办法

Class Class.forName(String packageNameAndClassName) throws ClassNotFoundException;    依据残缺的包名.类名获取对应的Class类对象    ClassNotFoundException 未找到指定类Class 类对象.getClass();    通过类对象获取以后类对象对应的Class类对象    例如:        Person p = new Person();  p.getClass() ==> Person类对应Class对象Class 类名.class;    通过类名获取以后类对应属性 Class对象    例如:         Person.class ==> Person类对应Class对象。        
package com.qfedu.a_reflect;/** * Class类办法演示 *  * @author 期年之前ying@ * */public class GetClassObject {    public static void main(String[] args) throws ClassNotFoundException {        /*         * Class Class.forName(String packageNameAndClassName)          *             throws ClassNotFoundException;         */        Class cls1 = Class.forName("com.project.a_reflect.Person");                /*         * Class 类对象.getClass();         */        Person person = new Person();        Class cls2 = person.getClass();                /*         * Class 类名.class;         */        Class cls3 = Person.class;                /*                  *         不论是通过哪一种形式获取指定类的Class对象,都是同一个Class对象         * 因为以后Person类在以后程序中有且只占用一次代码区空间。         */        System.out.println("cls1 == cls2 : " + (cls1 == cls2));        System.out.println("cls2 == cls3 : " + (cls2 == cls3));        System.out.println("cls3 == cls1 : " + (cls3 == cls1));    }}

操作Constructor 构造方法类

通过Class类对象获取对应类的Constructor构造方法类对象

Constructor[] getConstructors();    获取以后Class对象对应类中所有非私有化构造方法类对象数组。    Constructor[] getDeclaredConstructors();    【暴力反射】    获取以后Class对象对应类中的所有构造方法类对象数组,包含私有化构造方法。    Constructor getConstructor(Class... parameterTypes);    获取以后Class对象中,指定参数数据类型的构造方法。获取的构造方法为非私有化构造方法    Class... parameterTypes         Class类型不定长参数,用于束缚以后构造方法对应的数据类型。    例如:        无参数构造方法        cls.getConstructor(); ==> Person();        两个参数构造方法(int, String)        cls.getConstructor(int.class, String.class) ==> Person(int, String)        Constructor getDeclaredConstructor(Class... parameterTypes);    【暴力反射】    获取以后Class对象中,指定数据类型的构造方法,包含私有化构造方法    例如:        获取私有化String类型构造方法        cls.getDeclaredConstructor(String.class) ==> private Person(String.class)

操作Constructor类对象创立对应类对象

Object newInstance(Object... parameters);    通过Constructor类对象,执行对应的构造方法,创立对应类对象    Object... 不定长参数,要求数据类型为Object类型。    例如:        Person(); 无参数构造方法        Person p1 = (Person) constructor.newInstance();        Person(int, java.lang.String);        Person p2 = (Person) constructor.newInstance(10, "Java真好学"); 
package com.qfedu.a_reflect;import java.lang.reflect.Constructor;import java.lang.reflect.InvocationTargetException;/** * 操作Constructor构造方法类对象 *  * @author 期年之前ying@ * */public class GetConstructorObject {    public static void main(String[] args)             throws ClassNotFoundException, NoSuchMethodException, SecurityException,             InstantiationException, IllegalAccessException, IllegalArgumentException,             InvocationTargetException {        /*         * Class Class.forName(String packageNameAndClassName)          *             throws ClassNotFoundException;         */        Class cls = Class.forName("com.project.a_reflect.Person");                /*         * 1. 获取以后Class对象对应类中所有非私有化构造方法类对象数组         */        Constructor[] constructors = cls.getConstructors();        for (Constructor constructor : constructors) {            System.out.println(constructor);        }                System.out.println();                /*         * 2. 【暴力反射】         * 获取以后Class对象对应类中的所有构造方法类对象数组,包含私有化构造方法。         */        Constructor[] declaredConstructors = cls.getDeclaredConstructors();        for (Constructor constructor : declaredConstructors) {            System.out.println(constructor);        }                System.out.println();                /*         * 3. 获取以后Class对象中,指定参数数据类型的构造方法。获取的构造方法为非私有化构造方法         */        Constructor constructor1 = cls.getConstructor();        Constructor constructor2 = cls.getConstructor(int.class);        Constructor constructor3 = cls.getConstructor(int.class, String.class);        System.out.println(constructor1);        System.out.println(constructor2);        System.out.println(constructor3);                /*         * 4. 【暴力反射】             *    获取以后Class对象中,指定数据类型的构造方法,包含私有化构造方法         */        Constructor constructor4 = cls.getDeclaredConstructor(String.class);        System.out.println(constructor4);                System.out.println();        /*         * newInstance 创立类对象         */        Person p1 = (Person) constructor1.newInstance();        Person p2 = (Person) constructor2.newInstance(10);        Person p3 = (Person) constructor3.newInstance(20, "张三爱Java");        System.out.println(p1);        System.out.println(p2);        System.out.println(p3);                /*         * 给予暴力反射操作应用权限!!!         * setAccessible(boolean flag);         */        constructor4.setAccessible(true);        Person p4 = (Person) constructor4.newInstance("Java高兴多");        System.out.println(p4);    }}

操作 Method 成员办法类

通过Class类对象获取对应类的Method成员办法类对象

Method[] getMethods();    通过Class类对象调用,获取以后类内的所有非私有化成员办法,蕴含从父类继承而来子类能够应用的非私有化办法。    Method[] getDeclaredMethods();    【暴力反射】    通过Class类对象调用,获取以后类内的所有成员办法,包含私有化成员办法,然而不包含从父类继承而来的办法。    Method getMethod(String methodName, Class... parameterTypes);    通过Class类对象调用,依据办法名称和对应的形式参数列表数据类型获取对应的成员办法,能够获取父类继承办法,不能获取私有化成员办法    例如:        无参数成员办法 获取 game();            cls.getMethod("game");        有参数成员办法 获取 game(String);            cls.getMethod("game", String.class);    Method getDeclaredMethod(String methodName, Class... parameterTypes);    通过Class类对象调用,依据办法名称和对应的形式参数列表数据类型获取对应的成员办法,能够获取私有化成员办法,不能获取父类成员办法。    例如:        无参数私有化成员办法 testPrivate();            cls.getDeclaredMethod("testPrivate");        有参数私有化成员办法 testPrivate(String);            cls.getDeclaredMethod("testPrivate", String.class);

操作Method类对象执行办法

Object invoke(Object obj, Object... parameters);    通过Method类对象调用,执行对应办法。    Object obj 执行以后办法的类对象。    Object... parameters 对应以后办法的理论参数列表
package com.qfedu.a_reflect;import java.lang.reflect.InvocationTargetException;import java.lang.reflect.Method;/** * 操作Method类对象 *  * @author 期年之前ying@ * */public class GetMethodObject {    public static void main(String[] args)             throws ClassNotFoundException, SecurityException, NoSuchMethodException,             InstantiationException, IllegalAccessException, IllegalArgumentException,             InvocationTargetException {        /*         * Class Class.forName(String packageNameAndClassName)          *             throws ClassNotFoundException;         */        Class cls = Class.forName("com.project.a_reflect.Person");                /*         * 1. 通过Class类对象调用,获取以后类内的所有非私有化成员办法,         * 蕴含从父类继承而来子类能够应用的非私有化办法。         */        Method[] methods = cls.getMethods();        for (Method method : methods) {            System.out.println(method);        }                System.out.println();        /*         * 2. 获取以后类自有成员办法,包含私有化办法,然而不蕴含父类继承给子类的办法         */        Method[] declaredMethods = cls.getDeclaredMethods();        for (Method method : declaredMethods) {            System.out.println(method);        }        System.out.println();                /*         *  3. 依据指定办法名字和参数类型,获取非私有化成员办法         */        Method game1 = cls.getMethod("game");        Method game2 = cls.getMethod("game", String.class);                System.out.println(game1);        System.out.println(game2);        System.out.println();                /*         * 4. 依据指定的办法名称和参数类型,获取私有化成员办法         */        Method testPrivate1 = cls.getDeclaredMethod("testPrivate");        Method testPrivate2 = cls.getDeclaredMethod("testPrivate", String.class);        System.out.println(testPrivate1);        System.out.println(testPrivate2);        System.out.println();                /*         * 调用办法         */        Object object = cls.getConstructor().newInstance();        game1.invoke(object);        game2.invoke(object, "World Of Tank");                /*         * 给予暴力反射操作权限         */        testPrivate1.setAccessible(true);        testPrivate2.setAccessible(true);        testPrivate1.invoke(object);        testPrivate2.invoke(object, "西红柿+黄瓜+鸡蛋+羊肉串");    }}

操作 Field 成员变量类

通过Class类对象获取对应类的Field成员变量类对象

Field[] getFields();    获取类内所有非私有化成员变量数组    Field[] getDeclaredFields();    【暴力反射】    获取类内所有成员变量数组,包含私有化成员变量Field getField(String fieldName);    依据成员变量名字获取对应的成员变量对象,要求以后成员变量非私有化    例如:            public int test;        cls.getField("test");Field getDeclaredField(String fieldName);    【暴力反射】    获取类内指定名字的成员变量对象,包含私有化成员变量    例如:        private String name;        private int id;        cls.getDeclaredField("name");        cls.getDeclaredField("id");        

操作Field类对象赋值取值成员变量

Field[] getFields();    获取类内所有非私有化成员变量数组    Field[] getDeclaredFields();    【暴力反射】    获取类内所有成员变量数组,包含私有化成员变量Field getField(String fieldName);    依据成员变量名字获取对应的成员变量对象,要求以后成员变量非私有化    例如:            public int test;        cls.getField("test");Field getDeclaredField(String fieldName);    【暴力反射】    获取类内指定名字的成员变量对象,包含私有化成员变量    例如:        private String name;        private int id;        cls.getDeclaredField("name");        cls.getDeclaredField("id");        
package com.qfedu.a_reflect;import java.lang.reflect.Field;import java.lang.reflect.InvocationTargetException;/** * 操作Field类对象 *  * @author 期年之前ying@ * */public class GetFieldObject {    public static void main(String[] args)             throws ClassNotFoundException, NoSuchFieldException, SecurityException,            InstantiationException, IllegalAccessException, IllegalArgumentException,             InvocationTargetException, NoSuchMethodException {        /*         * Class Class.forName(String packageNameAndClassName)          *             throws ClassNotFoundException;         */        Class cls = Class.forName("com.project.a_reflect.Person");                /*         * 1. 获取类内所有非私有化成员变量数组         */        Field[] fields = cls.getFields();        for (Field field : fields) {            System.out.println(field);        }        System.out.println();                /*         * 2. 获取类内所有成员变量数组,包含私有化成员变量         */        Field[] declaredFields = cls.getDeclaredFields();        for (Field field : declaredFields) {            System.out.println(field);        }        System.out.println();                /*         * 3. 依据成员变量名字获取对应的成员变量对象,要求以后成员变量非私有化         */        Field test = cls.getField("test");        System.out.println(test);        System.out.println();                /*         * 4. 获取类内指定名字的成员变量对象,包含私有化成员变量         */        Field id = cls.getDeclaredField("id");        Field name = cls.getDeclaredField("name");        System.out.println(id);        System.out.println(name);            System.out.println();                /*         * 取值赋值成员变量         */        Object obj = cls.getConstructor().newInstance();        System.out.println(obj);        test.set(obj, 100);        System.out.println(obj);        System.out.println(test.get(obj));                id.setAccessible(true);        name.setAccessible(true);                id.set(obj, 10);        name.set(obj, "大哥好英武");        System.out.println(obj);        System.out.println(id.get(obj));        System.out.println(name.get(obj));                System.out.println();        System.out.println(id.getType());        System.out.println(name.getType());    }}

暴力反射受权

class AccessibleObject 类内办法public static void setAccessible(AccessibleObject[] array, boolean flag);    通过类名调用的动态工具形式,给予AccessibleObject类对象或者其子类对象数组,赋值操作权限。    子类对象包含: Field Method Constructor    public void setAccessible(boolean flag);    通过AccessibleObject类对象调用,繁多权限受权,Field Method Constructor都能够应用。

案例操作

须要应用    1. String办法    2. IO流 举荐字符流操作    3. 反射    4. 自行理解 ==> String 转其余类型办法 百度 parse系列办法
文件名:    studentInfo.txt文件内容:className=com.qfedu.a_reflect.Studentname=李四age=18gender=falsejavaScore=59webScore=59dbScore=59指标    文件内容转Student类对象
package com.qfedu.a_reflect;import java.io.BufferedReader;import java.io.FileReader;import java.io.IOException;import java.lang.reflect.Field;import java.lang.reflect.InvocationTargetException;import java.util.Arrays;@SuppressWarnings("all")public class ReflectDemo {    public static void main(String[] args)             throws IOException, ClassNotFoundException, InstantiationException,             IllegalAccessException, IllegalArgumentException, InvocationTargetException,            NoSuchMethodException, SecurityException, NoSuchFieldException {        // 1. 创立缓冲字符输出流 解决 文件        BufferedReader br = new BufferedReader(new FileReader("./data/studentInfo.txt"));                // 2. 读取文件数据        String classInfo = br.readLine();        String className = classInfo.substring(classInfo.indexOf("=") + 1);                // 3. 启动万恶之源 获取Class对象,加载指定类        Class cls = Class.forName(className);                // 4. 创立对应类对象        Object obj = cls.getConstructor().newInstance();                // 5. 读取文件,利用循环操作        String info = null;        Object value = null;                // 每一次从文件中读取一行数据        while ((info = br.readLine()) != null) {            // 依照 = 宰割信息 name=李四            String[] split = info.split("=");            System.out.println(Arrays.toString(split));            // 依据信息获取对应成员变量对象            Field field = cls.getDeclaredField(split[0]);            field.setAccessible(true);                        // 获取成员变量数据类型            Class type = field.getType();                        // 以后成员变量数据为String类型            if (type.equals(String.class)) {                value = split[1];                // field.set(obj, split[1]);            // 成员变量数据类型为int类型            } else if (type.equals(int.class)) {                value = Integer.parseInt(split[1]);            // 成员变量数据类型为boolean类型            } else if (type.equals(boolean.class)) {                value = Boolean.parseBoolean(split[1]);            }                        field.set(obj, value);        }                System.out.println(obj);                // 敞开资源        br.close();     }}

最初

欢送关注公众号:前程有光,支付一线大厂Java面试题总结+各知识点学习思维导+一份300页pdf文档的Java外围知识点总结! 这些材料的内容都是面试时面试官必问的知识点,篇章包含了很多知识点,其中包含了有基础知识、Java汇合、JVM、多线程并发、spring原理、微服务、Netty 与RPC 、Kafka、日记、设计模式、Java算法、数据库、Zookeeper、分布式缓存、数据结构等等。