反射整理学习
目录
1、反射介绍
2、反射API
2.1 获取类对应的字节码的对象(三种)
2.2 常用方法
3、反射的应用
3.1 创建 : 测试物料类
3.2 获取类对象
3.3 获取成员变量
3.4 通过字节码对象获取类的成员方法
3.5 通过字节码对象获取类的构造方法
4、创建对象
1、反射介绍
Reflection(反射) 是 Java 程序开发语言的特征之一,它允许运行中的 Java 程序对自身进行检查。被private封装的资源只能类内部访问,外部是不行的,但反射能直接操作类私有属性。反射可以在运行时获取一个类的所有信息,(包括成员变量,成员方法,构造器等),并且可以操纵类的字段、方法、构造器等部分。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法。所以先要获取到每一个字节码文件对应的Class类型的对象。
反射就是把java类中的各种成分映射成一个个的Java对象。
例如:一个类有:成员变量、方法、构造方法、包等等信息,利用反射技术可以对一个类进行解剖,把一个个组成部分映射成一个个对象。(其实:一个类中这些成员方法、构造方法、在加入类中都有一个类来描述)
加载的时候:Class对象的由来是将 .class 文件读入内存,并为之创建一个Class对象。
Class类
Class 类的实例表示正在运行的 Java 应用程序中的类和接口。也就是jvm中有N多的实例每个类都有该Class对象。(包括基本数据类型)
Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的defineClass 方法自动构造的。也就是这不需要我们自己去处理创建,JVM已经帮我们创建好了。
我们知道Spring框架可以帮我们创建和管理对象。需要对象时,我们无需自己手动new对象,直接从Spring提供的容器中的Beans获取即可。Beans底层其实就是一个Map<String,Object>,最终通过getBean(“user”)来获取。而这其中最核心的实现就是利用反射技术。
Bean
1、Java面向对象,对象有方法和属性,那么就需要对象实例来调用方法和属性(即实例化);
2、凡是有方法或属性的类都需要实例化,这样才能具象化去使用这些方法和属性;
3、规律:凡是子类及带有方法或属性的类都要加上注册Bean到Spring IoC的注解;(@Component , @Repository , @ Controller , @Service , @Configration)
4、把Bean理解为类的代理或代言人(实际上确实是通过反射、代理来实现的),这样它就能代表类拥有该拥有的东西了
5、在Spring中,你标识一个@符号,那么Spring就会来看看,并且从这里拿到一个Bean(注册)或者给出一个Bean(使用)
2、反射API
2.1 获取类对应的字节码的对象(三种)
① 调用某个类的对象的getClass()方法,即:对象.getClass();
Person p = new Person();
Class clazz = p.getClass();
注意:此处使用的是Object类中的getClass()方法,因为所有类都继承Object类,所以调用Object类中的getClass()方法来获取。
② 调用类的class属性类获取该类对应的Class对象,即:类名.class
Class clazz = Person.class;
③ 使用Class类中的forName()静态方法(最安全,性能最好)即:Class.forName(“类的全路径”)
Class clazz = Class.forName("类的全路径");
注意:在运行期间,一个类,只有一个Class对象产生。
三种方式常用第三种,第一种对象都有了还要反射干什么。第二种需要导入类的包,依赖太强,不导包就抛编译错误。
2.2 常用方法
当我们获得了想要操作的类的Class对象后,可以通过Class类中的方法获取和查看该类中的方法和属性。
//获取包名、类名clazz.getPackage().getName()//包名clazz.getSimpleName()//类名clazz.getName()//完整类名//获取成员变量定义信息getFields()//获取所有公开的成员变量,包括继承变量getDeclaredFields()//获取本类定义的成员变量,包括私有,但不包括继承的变量getField(变量名)getDeclaredField(变量名)//获取构造方法定义信息getConstructor(参数类型列表)//获取公开的构造方法getConstructors()//获取所有的公开的构造方法getDeclaredConstructors()//获取所有的构造方法,包括私有getDeclaredConstructor(int.class,String.class)//获取方法定义信息getMethods()//获取所有可见的方法,包括继承的方法getMethod(方法名,参数类型列表)getDeclaredMethods()//获取本类定义的的方法,包括私有,不包括继承的方法getDeclaredMethod(方法名,int.class,String.class)//反射新建实例clazz.newInstance();//执行无参构造创建对象clazz.newInstance(222,"韦小宝");//执行有参构造创建对象clazz.getConstructor(int.class,String.class)//获取构造方法//反射调用成员变量clazz.getDeclaredField(变量名);//获取变量clazz.setAccessible(true);//使私有成员允许访问f.set(实例,值);//为指定实例的变量赋值,静态变量,第一参数给nullf.get(实例);//访问指定实例变量的值,静态变量,第一参数给null//反射调用成员方法Method m = Clazz.getDeclaredMethod(方法名,参数类型列表);m.setAccessible(true);//使私有方法允许被调用m.invoke(实例,参数数据);//让指定实例来执行该方法
3、反射的应用
3.1 测试物料类
创建包: com.reflection
创建类: Student.java*
package com.review;/*本类用于复习反射的物料类*/public class Student {//1.定义成员变量private String name;public int age;//2.给被封装属性提供get与set方法public String getName() {return name;}public void setName(String name) {this.name = name;}//3.生成本类的无参构造与全参构造public Student(){}public Student(String name, int age) {this.name = name;this.age = age;}//4.提供本类的普通方法public void play(){System.out.println("不玩游戏,学Java!");}public void sunDay(int n){System.out.println("卷起来,没有假!");}//5.为了查看学生对象的具体属性与属性值,重写toString()@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", age=" + age +'}';}}
3.2 获取类对象
由于经常使用的是第三种方法,所以以下使用第三种。
创建包: com.reflection
创建类: TestReflect.java
/*本类用于反射的测试*/public class TestReflect {//1.可以创建程序的入口函数main()--此处不用//2.通过单元测试方法,获取目标类Student对应的字节码对象@Testpublic void getClazz() throws ClassNotFoundException {//练习获取字节码对象的3种方式Class<?> clazz1 = Class.forName("com.review.Student");Class<?> clazz2 = Student.class;Class<?> clazz3 = new Student().getClass();//打印的是Student类对应的字节码对象System.out.println(clazz1);//class com.reflection.Student//获取Student类对应的字节码对象clazz1的名字System.out.println(clazz1.getName());//com.reflection.Student//通过Student类对应的字节码对象,获取Student类的类名System.out.println(clazz1.getSimpleName());//通过Student类对应的字节码对象,获取Student类对应的包对象System.out.println(clazz1.getPackage());//通过Student类对应的字节码对象,先获取Student类对应的包对象,再获取这个包对象的名字System.out.println(clazz1.getPackage().getName());}}
3.3 获取成员变量
/**本类用来测试反射*/public class TestReflect {//3.通过单元测试方法练习引用类型数组的定义与遍历@Testpublic void getStu() {//1.创建Student类的3个对象Student s1 = new Student("张三", 3);Student s2 = new Student("李四", 4);Student s3 = new Student("王五", 5);//2.创建数组将刚刚的3个对象存入数组中Student[] s = {s1, s2, s3};//3.直接打印数组,查看数组中的元素System.out.println(Arrays.toString(s));//4.遍历学生数组,拿到每一个学生对象,做进一步的操作for (Student stu : s) {//System.out.println(stu);stu.play();//通过遍历到的对象,执行play()System.out.println(stu.age);//通过遍历到的对象,打印age属性}}//4.通过单元测试方法,获取Student类中的成员变量@Testpublic void getFie() throws ClassNotFoundException {//1.获取Student类对应的字节码对象Class<?> clazz = Class.forName("com.review.Student");//2.通过Student类对应的字节码对象获取Student类中的成员变量们Field[] fs = clazz.getFields();//3.遍历数组,获取Student类中的每个成员变量的具体信息/*注意!目前成员变量的修饰符必须是public的才能获取到*/for(Field f : fs){System.out.println(f.getName());//通过本轮循环到的字段对象获取字段名System.out.println(f.getType());//通过本轮循环到的字段对象获取字段的类型}}}
3.4 通过字节码对象获取类的成员方法
/**本类用来测试反射*/public class TestReflect {//5.通过单元测试方法,获取Student类中的成员方法@Testpublic void getFunction() {//1.获取Student类对应的字节码对象Class<?> clazz = Class.forName("com.review.Student");//2.通过Student类对应的字节码对象获取Student类中的成员方法们Method[] ms = clazz.getMethods();//3.通过高效for循环遍历数组,拿到每一个方法对象for (Method m : ms) {System.out.println(m);//直接打印遍历到的方法对象System.out.println(m.getName());//通过方法对象获取方法名Class<?>[] pt = m.getParameterTypes();//通过方法对象获取方法所有参数的数组System.out.println(Arrays.toString(pt));//打印方法参数的数组}}}
3.5 通过字节码对象获取类的构造方法
/**本类用来测试反射*/public class TestReflect {//6.通过单元测试方法,获取Student类中的构造方法@Testpublic void getCons() {//1.获取字节码对象Class<?> clazz = Class.forName("com.review.Student");//2.通过字节码对象获取目标类Student的构造方法们Constructor<?>[] cs = clazz.getConstructors();//3.通过高效for循环遍历数组for(Constructor c : cs){System.out.println(c.getName());//打印本轮遍历到的构造方法的名字Class[] pt = c.getParameterTypes();//通过本轮遍历到的构造函数对象获取构造函数的参数类型System.out.println(Arrays.toString(pt));//打印参数类型}}}
4、创建对象
/**本类用来测试反射*/public class TestReflect {//7.通过单元测试方法,创建Student目标类的对象@Testpublic void getObject() throws Exception {//1.获取字节码对象Class<?> clazz = Class.forName("com.review.Student");//2.通过反射技术创建目标类的对象,注意抛出异常/*反射创建对象方案1:使用 目标类 的 无参构造 创建对象*/Object o = clazz.newInstance();System.out.println(o);//这一步已经获取到了对象Student{name='null', age=0}/*反射创建对象方案2:使用 目标类 的 全参构造 创建对象* 思路:* 1.先获取指定的构造函数对象,注意需要指定构造函数的参数,传入的是.class字节码对象* 2.通过刚刚获取到的构造函数对象创建Student目标类的对象,并且给对象的属性赋值* *///3.获取目标类中指定的全参构造Constructor<?> c = clazz.getConstructor(String.class, int.class);//System.out.println(c);//4.通过获取到的构造函数:创建对象 + 给对象的属性赋值Object o2 = c.newInstance("赵六", 6);System.out.println(o2);}}相关文章:
反射整理学习
目录 1、反射介绍 2、反射API 2.1 获取类对应的字节码的对象(三种) 2.2 常用方法 3、反射的应用 3.1 创建 : 测试物料类 3.2 获取类对象 3.3 获取成员变量 3.4 通过字节码对象获取类的成员方法 3.5 通过字节码对象获取类的构造方法 4、创建对象…...
JavaScript 运算规则详解
在 JavaScript 中,运算规则是非常重要的基础知识,了解这些规则可以帮助我们正确地编写代码并避免一些常见的错误。本教程将详细介绍 JavaScript 中的各种运算规则,包括基本运算符、类型转换、运算优先级等内容。 1. 基本运算符 JavaScript …...
C++篇 语 句
到目前为止,我们只见过两种语句: return 语句和表达式语句。根据语句对执行顺 序的影响,C 语言其余语句大多属于以下 3 大类。 选择语句: if 语句和 switch 语句。循环语句: while 语句, do...while 语句和…...
简洁的在线观影开源项目
公众号:【可乐前端】,每天3分钟学习一个优秀的开源项目,分享web面试与实战知识。 每天3分钟开源 hi,这里是每天3分钟开源,很高兴又跟大家见面了,今天介绍的开源项目简介如下: 仓库名࿱…...
VB超级模块函数VB读写记事本-防止乱码支持UTF-8和GB2312编码
Private Sub Command1_Click() Writein “C:\Users\Administrator\Desktop\1.txt”, “文本文内容” End Sub Private Sub Form_Load() Text1 ReadANSI(“C:\Users\Administrator\Desktop\1.txt”) Text2 ReadUTF8(“C:\Users\Administrator\Desktop\1.txt”) End Sub 写入…...
XSS靶场-DOM型初级关卡
一、环境 XSS靶场 二、闯关 1、第一关 先看源码 使用DOM型,获取h2标签,使用innerHTML将内容插入到h2中 我们直接插入<script>标签试一下 明显插入到h2标签中了,为什么不显示呢?看一下官方文档 尽管插入进去了࿰…...
【嵌入式高级C语言】10:C语言文件
文章目录 1 文件的概述1.1 文件分类(存储介质)1.2 磁盘文件分类(存储方式)1.3 二进制文件和文本文件的区别 2 文件缓冲区3 文件指针4 文件的API4.1 打开文件4.2 关闭文件4.3 重新定位流4.3.1 fseek4.3.2 ftell4.3.3 rewind 4.4 字…...
创建数据表
Oracle从入门到总裁:https://blog.csdn.net/weixin_67859959/article/details/135209645 如果要进行数据表的创建 create table 表名称 (列名称 类型 [DEFAULT 默认值 ] ,列名称 类型 [DEFAULT 默认值 ] ,列名称 类型 [DEFAULT 默认值 ] ,...列名称 类型 [DEFAULT 默认值 ] )…...
C语言字符串型常量
在C语言中,字符串型常量是由一系列字符组成的常量。字符串常量在C中以双引号(")括起来,例如:“Hello, World!”。字符串常量在C中是不可变的,也就是说,一旦定义,就不能修改其内…...
计算机网络 八股
计算机网络体系结构 OSI:物理层、数据链路层、网络层、运输层、会话层、表示层、应用层...
深入了解 Jetpack Compose 中的 Modifier
Jetpack Compose 是 Android 中用于构建用户界面的现代化工具包。其中,Modifier 是一个非常重要的概念,它允许我们对 UI 组件进行各种样式和布局的调整。在本篇博客中,我们将深入了解 Modifier,以及如何在 Compose 中使用它。 什…...
【数据库】聚合函数|group by分组|having|where|排序|函数 关键字的使用
目录 一、聚合函数 1、max() 2、min() 3、avg() 4、sum() 5、count() 二、group by 分组汇总 一般聚合函数配合着group by(分组)语句进行使用 把一组的数据放到一起,再配合聚合函数进行使用 三、having having语句 做筛选的 四、where和having的作用以及区…...
docker安装mongoDB及使用
一.mongodb是什么? MongoDB是一个NoSQL的非关系型数据库 ,支持海量数据存储,高性能的读写 1.mongo的体系结构 SQL术语/概念MongoDB术语/概念解释/说明databasedatabase数据库tablecollection数据库表/集合rowdocument数据记录行/文档colum…...
Linux 之五:权限管理(文件权限和用户管理)
1. 文件权限 在Linux系统中,文件权限是一个非常基础且重要的安全机制。它决定了用户和用户组对文件或目录的访问控制级别。 每个文件或目录都有一个包含9个字符的权限模式,这些字符分为三组,每组三个字符,分别对应文件所有者的权限…...
基于YOLOv8深度学习的葡萄病害智能诊断与防治系统【python源码+Pyqt5界面+数据集+训练代码】深度学习实战
《博主简介》 小伙伴们好,我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源,可关注公-仲-hao:【阿旭算法与机器学习】,共同学习交流~ 👍感谢小伙伴们点赞、关注! 《------往期经典推…...
MySQL 在聚合函数查询的结构中继续过滤
HAVING HAVING 关键字和 WHERE 关键字都可以用来过滤数据,且 HAVING 支持 WHERE 关键字中所有的操作符和语法,如果想要从 GROUP BY 分组中进行筛选的话,不是用 WHERE 而是使用 HAVING 来进行聚合函数的筛选。 语法 SELECT <列名1>, <列名2>,…...
UE4.27_ParticleSystem(没写完的材料)
UE4.27_ParticleSystem(没写完的材料) 参考实例: UE4[蓝图]下雪效果及雪的材质的实现...
腾讯云轻量服务器流量用完了怎么办?停机吗?
腾讯云轻量服务器流量用完了怎么办?超额流量另外支付流量费,流量价格为0.8元/GB,会自动扣你的腾讯云余额,如果你的腾讯云账号余额不足,那么你的轻量应用服务器会面临停机,停机后外网无法访问,继…...
块级作用域、变量提升
1.块级作用域 JS 中作用域有:全局作用域、函数作用域。没有块作用域的概念。ECMAScript 6(简称 ES6)中新增了块级作用域。块作用域由 { } 包括,if 语句和 for 语句里面的{ }也属于块作用域。 2.变量提升 如果变量声明在函数里面,则将变量声…...
c# 连接oracle 及对应获取数据集
1、数据库配置xml,首先连接成功后会自动创建xml并保存到对应xml,如下 static string ConnPath AppDomain.CurrentDomain.BaseDirectory "ConnOrcle.xml"; 声明xml名称,便于后续写入对应数据库参数 2、创建xml /// <summar…...
【根据当天日期输出明天的日期(需对闰年做判定)。】2022-5-15
缘由根据当天日期输出明天的日期(需对闰年做判定)。日期类型结构体如下: struct data{ int year; int month; int day;};-编程语言-CSDN问答 struct mdata{ int year; int month; int day; }mdata; int 天数(int year, int month) {switch (month){case 1: case 3:…...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...
系统设计 --- MongoDB亿级数据查询优化策略
系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log,共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题,不能使用ELK只能使用…...
dedecms 织梦自定义表单留言增加ajax验证码功能
增加ajax功能模块,用户不点击提交按钮,只要输入框失去焦点,就会提前提示验证码是否正确。 一,模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...
微信小程序 - 手机震动
一、界面 <button type"primary" bindtap"shortVibrate">短震动</button> <button type"primary" bindtap"longVibrate">长震动</button> 二、js逻辑代码 注:文档 https://developers.weixin.qq…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...
Android15默认授权浮窗权限
我们经常有那种需求,客户需要定制的apk集成在ROM中,并且默认授予其【显示在其他应用的上层】权限,也就是我们常说的浮窗权限,那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...
MySQL用户和授权
开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务: test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...
RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...
