Java基础——反射

反射 框架设计的灵魂

Java中的反射顾名思义就是将类的各个组成部分封装为其他对象。使用这些封装后的对象可以进行一些操作。
对于反射机制,我们可以说反射就是框架设计的灵魂。很多框架内部的机制都是反射。
使用反射的好处就是:

  • 可以在程序运行过程中,操作这些对象
  • 可以解耦,提高程序的可扩展性

Class

说起反射我们必须说一个重要的类那就是Class。这个类代表了Java编译后的字节码对象。字节码对象包含了定义类时所指定的全部的成员变量、方法等属性。

Java执行的三个阶段

Java代码的三个阶段.jpg

Java的执行过程是:我将Java代码的执行阶段分为3个阶段。分别是Source源代码阶段、Class类对象阶段、Runtime运行时阶段

  • Source源代码阶段:在这个阶段中,Java会将源代码编译为字节码文件,也就是我们平时见到的.class文件。
  • Class类对象阶段:在这个阶段中,当我们使用对应的对象时,类加载器ClassLoader就会将字节码文件对象加载进内存。Class对象中封装了源代码中成员变量、构造方法、成员方法。
  • Runtime运行时阶段:在这个阶段中在使用对应的类对象的时候,JVM会将对应的字节码文件也就是我们前面说过的Class对象加载进内存,这样我们就能使用定义好的类和对象了。

Class对象的方式

  1. Class.from("全类名") 将字节码文件加载进内存,返回一个class对象
  • 一般多用于配置文件,将类名定义在配置文件中,读取配置文件,加载类
  1. 类名.class 通过类名的属性class获取 class对象
  • 多用于参数的传递
  1. 对象.getClass():getClass()方法定义在超类Object中

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

Class对象功能

我们刚才说了通过Class对象我们可以操作对象的属性,那么我们怎么获取对象的属性呢

获取成员变量

  • Field[] getFields():获取所有public修饰的成员变量 返回一个Field数组
  • Field getField(String name) 获取指定名称的public 修饰的变量

使用getFields只能获取Public修饰的成员变量

  • Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符
  • Field getDeclaredField(String name) 获取指定名称的成员变量 不考虑修饰符

获取构造方法

  • Constructor<?>[] getConstructors() 获取构造方法的对象数组 public修饰
  • Constructor getConstructor(类<?>...parameterTypes)获取指定参数的构造方法 public修饰
  • Constructor getDeclaredConstructor(类<?>...parameterTypes) 获取构造方法对象 不考虑修饰符
  • Constructor[] getDeclaredConstructor() 获取构造方法的对象数组 不考虑修饰符

获取成员方法

  • Method[] getMethods() 获取成员方法数组 public修饰

getMethods不仅仅会获取类的成员方法,还会获取父类的方法

  • Method getMethod(String name,类<?>...parameterTypes) 获取指定参数和类型的成员方法
  • Method[] getDeclareMethods 获取成员方法数组 不考虑修饰符
  • Method getDeclareMethod(String name,类<?>....parameterTypes) 获取指定参数的成员方法 不考虑修饰符

获取全类名

  • String getName() 获取class对象的全类名

Field成员变量

我们获取了Field成员变量后可以通过Field设置对应的成员变量的值
1 设置值
* void set(Object obj,Object value)
2 获取值
* get(Object obj)
3 忽略访问权限修饰符的安全检查
* setAccessible(true) 暴力反射

Constructor 构造方法

通过Constructor构造方法对象。

  • T newInstance(Object...initargs) 创建对象
  • 如果使用空参数构造方法的创建对象,操作可以简化 Class对象的newInstance方法
     Person person2 = (Person) cla.newInstance();
            System.out.println(person2);

如果构造器是private修饰的,我们也可以调用construct的setAccessible来进行暴力反射
constructor.setAccessible(true) //暴力反射
暴力反射的前提必须使用declare的方法

Method方法对象

  • 执行方法
    • Object invoke(Object obj,Object...args)
  • 获取方法名称
    • String getName获取方法名
      普通的Method打印的是方法的全名(包名.类名.方法名)
      而getName获取的方法名就是方法的名称

学习了这么多,我们来实现一个案例来看一下反射的使用

案例

  • 需求:写一个小框架,在不改变任何类的情况下,执行类中的任意方法
  • 步骤:
  1. 将需要创建的类的全类名和要执行的方法定义在配置文件中
  2. 在程序中加载读取配置文件
  3. 使用反射技术来加载类文件进内存
  4. 创建对象
  5. 执行方法
  • 定义一个配置文件 pro.properties
className=com.probuing.bean.Student
methodName=study
  • 创建要执行的实体类对象 Student.java
public class Student {
    public void study() {
        System.out.println("this is student is studing");
    }
}
  • 创建执行框架
public class ReflectFrame {
    public static void main(String[] args) {
        try {
            //加载配置文件
            Properties pro = new Properties();
            //获取类加载器,获取配置文件路径
            ClassLoader classLoader = ReflectFrame.class.getClassLoader();
            InputStream resourceAsStream = classLoader.getResourceAsStream("pro.properties");
            //加载配置文件,转换为一个集合
            pro.load(resourceAsStream);
            //获取配置文件中定义的数据
            String className = pro.getProperty("className");
            String methodName = pro.getProperty("methodName");
            //加载指定的类class进内存
            Class<?> cla = Class.forName(className);
            //创建对象
            Student student = (Student) cla.newInstance();
            //获取方法对象
            Method method = cla.getMethod(methodName);
            //执行方法
            method.invoke(student);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,185评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,652评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,524评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,339评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,387评论 6 391
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,287评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,130评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,985评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,420评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,617评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,779评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,477评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,088评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,716评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,857评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,876评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,700评论 2 354

推荐阅读更多精彩内容

  • 一、类的加载 (一) 定义及过程: 当程序需要使用某个类的时候,如果这个类还没有被加载到内存中,则系统会通过加载、...
    VictorBXv阅读 320评论 0 2
  • 夯实 Java 基础 - 反射 自嵌套 Fragment 懒加载文章至今已经已经一个星期过去了,说实话最近对于学习...
    醒着的码者阅读 723评论 2 2
  • 反射注解动态代理相关阅读 Java基础:类加载器 Java基础:反射 Java基础:注解 Java基础:动态代理 ...
    JackChen1024阅读 924评论 3 27
  • 深入理解Class对象 RRTI的概念以及Class对象作用 认识Class对象之前,先来了解一个概念,RTTI(...
    架构师springboot阅读 1,561评论 0 3
  • JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的...
    陈晨_Fly阅读 960评论 0 20