反射:Web学习的灵魂

我们从最初的 javac -HelloWorld.java,到面向对象部分,我们可以将Java代码在计算机中经历的阶段分为三部分:Scource源代码阶段 —— Class类对象阶段 —— Runtime运行时阶段 而我们知道,Java中一个类在源代码阶段,是存储在硬盘中的,而编译后,就已经被加载到内存中区,那么有没有一种方法可以在这种情况下,获取或者修改它的方法或者属性呢?这就是我们今天所讲的Java反射机制

(一) 概述以及好处

(1) 概述

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。——百度百科

我们将类分为三部分,成员变量,构造方法,成员方法,代码编译后,变成了字节码文件(.class文件)而万物皆对象,所以在字节码文件中,又将这三部分分别整合成对象,所以我们得出结论:

反射:将类的各个组成部分封装成对象,并通过这个Class类型的对象,去使用该文件中的成员变量,构造方法,成员方法

(2) 好处

  1. 可以在程序“运行时”,对 .class文件进行操作,并且由此获取以及操作 类中的各个部分
  2. 可以解耦,提高程序的扩展性
  3. 增加程序的灵活性 (最后案例中体现)

(二) 获取Class对象的方式

(1)将字节码文件加载进内存,返回Class对象

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

  1. Class.forName("全类名")

(2)多用于参数的传递

  1. 类名.class

(3)getClass()方法在Object类中定义着

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

  1. 对象.getClass():

三种方式的代码实现

  1. package cn.ideal.reflect;
  2. import cn.ideal.domain.Student;
  3. public class ReflectDemo1 {
  4. public static void main(String[] args) throws Exception {
  5. //1.Class.forName("全类名")
  6. Class cls1 = Class.forName("cn.ideal.domain.Student");
  7. System.out.println(cls1);
  8. //2.类名.class
  9. Class cls2 = Student.class;
  10. System.out.println(cls2);
  11. //3.对象.getClass()
  12. Student p = new Student();
  13. Class cls3 = p.getClass();
  14. System.out.println(cls3);
  15. //用 == 比较
  16. System.out.println(cls1 == cls2);//true
  17. System.out.println(cls2 == cls3);//true
  18. }
  19. }
  20. //运行结果
  21. class cn.ideal.domain.Student
  22. class cn.ideal.domain.Student
  23. class cn.ideal.domain.Student
  24. true
  25. true

通过上面的案例我们可以得出:

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

(三) Class对象功能

(1) 获取功能:

获取成员变量们
  1. //获取所有public修饰的成员变量
  2. Field[] getFields()
  3. //获取指定名称的public修饰的成员变量
  4. Field getField(String name)
  5. //获取所有的成员变量,不考虑修饰符
  6. Field[] getDeclaredFields()
  7. Field getDeclaredField(String name)
获取构造方法们
  1. Constructor<?>[] getConstructors()
  2. Constructor<T> getConstructor(类<?>... parameterTypes)
  3. Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
  4. Constructor<?>[] getDeclaredConstructors()

获取成员方法们

  1. Method[] getMethods()
  2. Method getMethod(String name, 类<?>... parameterTypes)
  3. Method[] getDeclaredMethods()
  4. Method getDeclaredMethod(String name, 类<?>... parameterTypes)

获取全类名

  1. String getName()

Field:成员变量

  1. 操作:
  2. //设置值
  3. void set(Object obj, Object value)
  4. //获取值
  5. get(Object obj)
  6. //忽略访问权限修饰符的安全检查
  7. setAccessible(true):暴力反射

Constructor:构造方法

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

Method:方法对象

  1. //执行方法
  2. Object invoke(Object obj, Object... args)
  3. //获取方法名称
  4. String getName:获取方法名

我们来具体使用一下上面的一些方法

  1. package cn.ideal.reflect;
  2. import cn.ideal.domain.Student;
  3. import java.lang.reflect.Field;
  4. public class ReflectDemo2 {
  5. public static void main(String[] args) throws Exception {
  6. //获取Student的Class对象
  7. Class studentClass = Student.class;
  8. /*
  9. 1. 获取成员变量们
  10. * Field[] getFields()
  11. * Field getField(String name)
  12. * Field[] getDeclaredFields()
  13. Field getDeclaredField(String name)
  14. */
  15. //Field[] getFields() 获取所有public修饰的成员变量
  16. Field[] fields = studentClass.getFields();
  17. for (Field field : fields) {
  18. System.out.println(field);
  19. }
  20. System.out.println("----------");
  21. //Field getField(String name)
  22. Field a = studentClass.getField("a");
  23. //获取成员变量a的值
  24. Student s = new Student();
  25. Object value = a.get(s);
  26. System.out.println(value);
  27. //设置a的值
  28. a.set(s, "张三");
  29. System.out.println(s);
  30. System.out.println("----------");
  31. //Field[] getDeclaredFields() 获取所有的成员变量
  32. Field[] declaredFields = studentClass.getDeclaredFields();
  33. for (Field declaredField : declaredFields) {
  34. System.out.println(declaredField);
  35. }
  36. System.out.println("----------");
  37. //Field getDeclaredField(String name)
  38. Field d = studentClass.getDeclaredField("d");
  39. //忽略访问权限修饰符
  40. d.setAccessible(true);//暴力反射
  41. Object value2 = d.get(s);
  42. System.out.println(value2);
  43. }
  44. }
  45. //运行结果
  46. public java.lang.String cn.ideal.domain.Student.a
  47. ----------
  48. null
  49. Student{name='null', age=0, a='张三', b='null', c='null', d='null'}
  50. ----------
  51. private java.lang.String cn.ideal.domain.Student.name
  52. private int cn.ideal.domain.Student.age
  53. public java.lang.String cn.ideal.domain.Student.a
  54. protected java.lang.String cn.ideal.domain.Student.b
  55. java.lang.String cn.ideal.domain.Student.c
  56. private java.lang.String cn.ideal.domain.Student.d
  57. ----------
  58. null
  1. package cn.ideal.reflect;
  2. import cn.ideal.domain.Student;
  3. import java.lang.reflect.Constructor;
  4. public class ReflectDemo3 {
  5. public static void main(String[] args) throws Exception {
  6. //获取Student的Class对象
  7. Class studentClass = Student.class;
  8. /*
  9. 2. 获取构造方法们
  10. * Constructor<?>[] getConstructors()
  11. * Constructor<T> getConstructor(类<?>... parameterTypes)
  12. * Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
  13. * Constructor<?>[] getDeclaredConstructors()
  14. */
  15. //Constructor<T> getConstructor(类<?>... parameterTypes)
  16. Constructor constructor = studentClass.getConstructor(String.class, int.class);
  17. System.out.println(constructor);
  18. //创建对象 带参
  19. Object student = constructor.newInstance("张三", 20);
  20. System.out.println(student);
  21. System.out.println("----------");
  22. Constructor constructor1 = studentClass.getConstructor();
  23. System.out.println(constructor1);
  24. //创建对象 不带参
  25. Object student1 = constructor1.newInstance();
  26. System.out.println(student1);
  27. //创建对象 不带参 (推荐方法)
  28. Object o = studentClass.newInstance();
  29. System.out.println(o);
  30. }
  31. }
  1. package cn.ideal.reflect;
  2. import cn.ideal.domain.Student;
  3. import java.lang.reflect.Method;
  4. public class ReflectDemo4 {
  5. public static void main(String[] args) throws Exception {
  6. //获取Student的Class对象
  7. Class studentClass = Student.class;
  8. /*
  9. 3. 获取成员方法们:
  10. * Method[] getMethods()
  11. * Method getMethod(String name, 类<?>... parameterTypes)
  12. * Method[] getDeclaredMethods()
  13. * Method getDeclaredMethod(String name, 类<?>... parameterTypes)
  14. */
  15. //获取指定名称的方法
  16. Method study_method = studentClass.getMethod("study");
  17. Student s = new Student();
  18. //执行方法
  19. study_method.invoke(s);
  20. Method study_method2 = studentClass.getMethod("study", String.class);
  21. //执行方法
  22. study_method2.invoke(s, "英语");
  23. System.out.println("----------");
  24. //获取所有public修饰的方法
  25. Method[] methods = studentClass.getMethods();
  26. for (Method method : methods) {
  27. System.out.println(method);
  28. String name = method.getName();
  29. System.out.println(name);
  30. }
  31. //获取类名
  32. String className = studentClass.getName();
  33. System.out.println(className);
  34. }
  35. }

(四) 反射案例

通过修改配置文件,达到不改变该类的任何代码,可以创建任意类的对象,可以执行任意方法,避免了每一次都修改类文件的缺点,同时提高了程序的扩展性

  1. * 实现:
  2. 1. 配置文件
  3. 2. 反射
  4. * 步骤:
  5. 1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
  6. 2. 在程序中加载读取配置文件
  7. 3. 使用反射技术来加载类文件进内存
  8. 4. 创建对象
  9. 5. 执行方法
  1. //pro.properties 自定义配置文件
  2. className=cn.ideal.domain.Student
  3. methodName=study
  1. package cn.ideal.reflect;
  2. import java.io.InputStream;
  3. import java.lang.reflect.Method;
  4. import java.util.Properties;
  5. public class ReflectTest {
  6. public static void main(String[] args) throws Exception {
  7. /*
  8. 不改变该类的任何代码,可以创建任意类的对象,可以执行任意方法
  9. */
  10. //1.加载配置文件
  11. //创建Properties对象
  12. Properties pro = new Properties();
  13. //加载配置文件,转换为一个集合
  14. //获取class目录下的配置文件
  15. ClassLoader classLoader = ReflectTest.class.getClassLoader();
  16. InputStream is = classLoader.getResourceAsStream("pro.properties");
  17. pro.load(is);
  18. //2.获取配置文件中定义的数据
  19. String className = pro.getProperty("className");
  20. String methodName = pro.getProperty("methodName");
  21. //3.加载该类进内存
  22. Class cls = Class.forName(className);
  23. //4.创建对象
  24. Object obj = cls.newInstance();
  25. //5.获取方法对象
  26. Method method = cls.getMethod(methodName);
  27. //6.执行方法
  28. method.invoke(obj);
  29. }

我仅仅浅薄的谈了一下反射的基本知识,一些比较深入的理解由于篇幅问题,放在后面专篇讲解,感谢支持。确实有很多不足之处,也希望能与大家交流。

结尾:

如果内容中有什么不足,或者错误的地方,欢迎大家给我留言提出意见, 蟹蟹大家 !_

如果能帮到你的话,那就来关注我吧!(系列文章均会在公众号第一时间更新)

在这里的我们素不相识,却都在为了自己的梦而努力 ❤

一个坚持推送原创Java技术的公众号:理想二旬不止

Java反射:Web学习的灵魂的更多相关文章

  1. JAVA反射机制—学习总结

    最近收到很多关于Java反射机制的问题留言,其实Java反射机制技术方面没有太多难点,或许是大家在学习过程中遗漏了细小知识点,导致一些问题无法彻底理解,现在我们简单的总结一下,加深印象.什么是反射机制 ...

  2. 云笔记项目-Java反射知识学习

    在云笔记项目中,补充了部分反射的知识,反射这一部分基础知识非常重要,前面学习的框架Spring和MyBatis读取xml配置文件创建对象,以及JDBC加载驱动等都用了反射,但只知道有这个东西,具体不知 ...

  3. Java反射机制——学习总结

    前几天上REST课,因为涉及到Java的反射机制,之前看过一直没有用过,有些遗忘了,周末找了些资料来重新学习,现在总结一下,加深印象. 什么是反射机制? 参考百度百科对java反射机制的定义: “JA ...

  4. Java反射机制学习与研究

    Java反射机制:可以获取正在运行时的Java对象. 1.判断运行时对象对象所属的类. 2.判断运行时对象所具有的成员变量和方法. 3.还可以调用到private方法,改变private变量的值. S ...

  5. Java 反射机制学习资料

    Java反射——引言 Java反射——Class对象 Java反射——构造函数 Java反射——字段 Java反射——方法 Java反射——Getter和Setter Java反射——私有字段和私有方 ...

  6. Java反射机制学习

    Java 反射是Java语言的一个很重要的特征,它使得Java具体了“动态性”. 在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法?答 ...

  7. JAVA反射机制学习随笔

    JAVA反射机制是用于在运行时动态的获取类的信息或者方法,属性,也可以用来动态的生成类,由于所有类都是CLASS的子类,我们可以用一个CLASS类的实例来实例化各种类 例如: Class<?&g ...

  8. java反射机制学习代码

    根据 http://www.iteye.com/topic/137944 文档进行学习 代码如下: package reflectTest; import java.lang.reflect.*; i ...

  9. Java反射篇学习笔记

    今天重新学习了java中的反射,写一篇学习笔记总结一下.代码基本都是照着两篇博客敲的: 参考一:   https://blog.csdn.net/sinat_38259539/article/deta ...

随机推荐

  1. oracle数据库外连接

    外连接作用:(左外连接和右外连接;注:没有全外连接) 希望把某些不成立的记录(40号部门),仍然包含在最后的结果中 左外连接:当where e.deptno=d.deptno不成立的时候,等号左边的表 ...

  2. CF788A Functions again dp

    求一个最长子段和就完了,可以出T1? code: #include <bits/stdc++.h> #define N 100006 #define ll long long #defin ...

  3. [POI2008]BLO-Blockade 割点

    [POI2008]BLO-Blockade 割点 题面 容易想到用\(\text{Tarjan}\)求割点.对于非割点,会损失\(2\times(n-1)\)次访问(注意是互相访问,所以要乘2):对于 ...

  4. 【概率论】4-1:随机变量的期望(The Expectation of a Random Variable Part I)

    title: [概率论]4-1:随机变量的期望(The Expectation of a Random Variable Part I) categories: - Mathematic - Prob ...

  5. 解决node-sass无法下载的问题

    本文链接:https://blog.csdn.net/qq383366204/article/details/86605960在国内用npm安装依赖的时候经常都会有各种奇怪的问题,个人强烈推荐用yar ...

  6. Jmeter linux 运行

    一.在Linux服务器先安装sdk 1.先从客户端下载jdk1.8.0_144.tar.gz,再上传到服务器 2.解压:tar -xzf jdk1.8.0_144.tar.gz,生成文件夹 jdk1. ...

  7. UVALive 3716 DNA Regions ——(扫描法)

    乍一看这个问题似乎是很复杂,但其实很好解决. 先处理出每个点到原点的距离和到x正半轴的角度(从x正半轴逆时针旋转的角度).然后以后者进行排序. 枚举每一个点到圆心的距离,作为半径,并找出其他到圆心距离 ...

  8. Redis集群都有哪些模式

    前言: 一,为什么要使用redis 1,解决应用服务器的cpu和内存压力 2,减少io的读操作,减轻io的压力 3,关系型数据库扩展性不强,难以改变表的结构 二,优点 1,nosql数据库没有关联关系 ...

  9. linux 后台 运行

    但是如果终端关闭的话,程序也会终止,那么就要涉及到linux的一个十分强大的命令:screen. 按照我个人的理解,这个命令就是能够在linux中创造出多个终端,在已有的窗口内部再创造更多的窗口,结合 ...

  10. jinja2-宏,include, import

    一 宏 宏类似常规编程语言中的函数.它们用于把常用行为作为可重用的函数,取代 手动重复的工作.如果宏在不同的模板中定义,你需要首先使用 import,比如 {% macro input(name, v ...