day42-反射01
Java反射01
1.反射(reflection)机制
1.1反射机制问题
一个需求引出反射
请看下面问题:
- 根据配置文件 re.properties 指定信息,创建Cat对象并调用方法hi
classfullpath=li.reflection.Cat
method=hi
使用现有的技术,你能做的到吗?
这样的需求在学习框架时特别多,即通过外部文件配置,在不修改源码的情况下来控制程序,也符合设计模式的ocp原则(开闭原则)
开闭原则:不修改源码,扩展功能
例子:
re.properties:
classfullpath=li.reflection.Cat
method=cry
Cat:
package li.reflection;
public class Cat {
private String name = "招财猫";
public void hi() {
System.out.println("hi " + name);
}
public void cry() {
System.out.println(name + " cry");
}
}
ReflectionQuestion:
package li.reflection;
import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Properties;
public class ReflectionQuestion {
public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
//根据配置文件 re.properties 指定信息,创建Cat对象 并调用方法hi
//1.传统方法 new 对象 -->调用方法
//Cat cat = new Cat();
//cat.hi();
//2.尝试使用读取文件的方法
//2.1使用properties类,可以读写配置文件
Properties properties = new Properties();
properties.load(new FileInputStream("src\\re.properties"));
String classfullpatch = properties.get("classfullpath").toString();
String methodName = properties.get("method").toString();
System.out.println("classfullpath" + classfullpatch);
System.out.println("method" + methodName);
//2.2创建对象
//使用传统的方法行不通
//3.使用反射机制解决
//3.1加载类,返回一个Class类型的对象cls(这里的Class就是一个类,他的类名就叫Class)
Class cls = Class.forName(classfullpatch);
//3.2通过cls得到加载的类 li.reflection.Cat 的一个对象实例
Object o = cls.newInstance();
System.out.println("o的运行类型=" + o.getClass());//o的运行类型
//3.3通过 cls 得你加载的类 li.reflection.Cat 的methodName"hi" 的方法对象
// 即:在反射机制中,可以把方法视为一个对象(万物皆对象)
Method method1 = cls.getMethod(methodName);
//3.4通过 method1 调用方法:即通过方法对象来实现调用方法
System.out.println("========");
method1.invoke(o);//传统方法: 对象.方法() , 反射机制:方法.invoke(对象)
//意义在与反射机制可以通过不求该源码就完成功能的拓展
/**
* 例如,在Cat类中有两个方法,hi()和cry(),现在要求将调用用的hi方法改为调用cry方法,
* 这时候只需要在配置文件re.properties中修改引用的方法名即可
* 不需像想传统方法一样,需要在源码中修改
*/
}
}

1.2反射机制
1.2.1Java Reflection
- 反射机制允许程序在执行期间借助于Reflection API取得任何类的内部信息(比如成员变量,构造器,成员方法等等),并能够操作对象的属性以及方法。反射在设计模式和框架底层都会用到
- 加载完类之后,在堆中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象包含了类的完整结构信息。通过这个对象得到类的结构。这个Class对象就像一面镜子,透过这个镜子看到了的结构,所以形象地称之为:反射
例如:一个Person类型的实例对象叫 p
则 p 对象 - -> 类型 Person类
Class 对象 cls --> 类型 Class 类(这个类的名字就叫Class)
1.2.2Java反射机制原理图

- Java反射机制可以完成
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时得到任意一个类所具有的成员变量和方法
- 在运行时调用任意一个对象的成员变量和方法
- 生成动态代理
1.2.3反射相关类
反射相关的主要类:
- java.lang.Class:代表一个类,Class对象表示某个类加载过后在堆中的对象
- java.lang.reflect.Method:表示类的方法(Method对象表示某个类的方法)
- java.lang.reflect.Field:表示类的成员变量(Field对象表示某个类的成员变量)
- java.lang.reflect.Constructor:表示类的构造方法(Constructor对象表示某个类的构造器)
这些类在 java.lang.reflect
例子:
Cat:
package li.reflection;
public class Cat {
private String name = "招财猫";
public int age = 10;
public Cat() {
}//无参构造器
public Cat(String name) {
this.name = name;
}
public void hi() {
System.out.println("hi " + name);
}
public void cry() {
System.out.println(name + " cry");
}
}
re.properties:
classfullpath=li.reflection.Cat
method=hi
Reflection01:
package li.reflection;
import java.io.FileInputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Properties;
public class Reflection01 {
public static void main(String[] args) throws Exception {
//使用properties类,读写配置文件
Properties properties = new Properties();
properties.load(new FileInputStream("src\\re.properties"));
String classfullpatch = properties.get("classfullpath").toString();
String methodName = properties.get("method").toString();
System.out.println("classfullpath" + classfullpatch);
System.out.println("method" + methodName);
//反射机制
//1.加载类,返回一个Class类型的对象cls(这里的Class就是一个类,他的类名就叫Class)
Class cls = Class.forName(classfullpatch);
//通过cls得到加载的类 li.reflection.Cat 的一个对象实例
Object o = cls.newInstance();
System.out.println("o的运行类型=" + o.getClass());//o的运行类型
//2.通过 cls得到加载的类 li.reflection.Cat 的methodName"hi" 的方法对象
// 即:在反射机制中,可以把方法视为一个对象(万物皆对象)
Method method1 = cls.getMethod(methodName);
//通过 method1 调用方法:即通过方法对象来实现调用方法
System.out.println("========");
method1.invoke(o);//传统方法: 对象.方法() , 反射机制:方法.invoke(对象)
//3.java.lang.reflect.Field:表示类的成员变量(Field对象表示某个类的成员变量)
//得到name字段
//getField不能得到私有的属性
Field nameField = cls.getField("age");
System.out.println(nameField.get(o));//传统方法:对象.成员变量 , 反射:成员变量的对象.get(对象)
//4.java.lang.reflect.Constructor:表示类的构造方法(Constructor对象表示某个类的构造器)
Constructor constructor1 = cls.getConstructor();//()中可以指定构造器的类型,这里返回无参构造器
System.out.println(constructor1);//Cat()
Constructor constructor2 = cls.getConstructor(String.class);//这里传入的String.class 就是String类的Class对象
System.out.println(constructor2);//Cat(String name)
}
}

1.2.4反射优点和缺点
- 优点:可以动态地创建和使用对象(也是底层框架的核心),使用灵活,没有反射机制,框架技术就失去底层支撑。
- 缺点:使用反射机制基本是解释执行,对执行速度有影响
- 调用反射优化-关闭访问检查
- Method和Field、Constructor对象都有setAccessible()方法
- setAccessible作用是启动和禁用服务安全检查的开关
- 参数值为true表示 反射的对象在使用时 取消访问检查,提高反射效率。参数值为false则表示反射的对象执行访问检查

例子:测试反射调用的性能 和优化方案
package li.reflection;
import java.lang.reflect.Method;
//测试反射调用的性能 和 优化方案
public class Reflection02 {
public static void main(String[] args) throws Exception {
m1();
m2();
m3();
}
//传统方法,调用hi
public static void m1() {
Cat cat = new Cat();
long start = System.currentTimeMillis();
for (int i = 0; i < 900000000; i++) {
cat.hi();
}
long end = System.currentTimeMillis();
System.out.println("m1() 耗时=" + (end - start));
}
//反射机制调用hi
public static void m2() throws Exception {
//拿到Cat类的Class对象
Class cls = Class.forName("li.reflection.Cat");//这里为了方便,就不读文件了
//构造Cat类的对象
Object o = cls.newInstance();
//得到Cat类的成员方法
Method hi = cls.getMethod("hi");
long start = System.currentTimeMillis();
for (int i = 0; i < 900000000; i++) {
hi.invoke(o);//反射机制调用方法
}
long end = System.currentTimeMillis();
System.out.println("m2() 耗时=" + (end - start));
}
//反射调用优化
public static void m3() throws Exception {
//拿到Cat类的Class对象
Class cls = Class.forName("li.reflection.Cat");//这里为了方便,就不读文件了
//构造Cat类的对象
Object o = cls.newInstance();
//得到Cat类的成员方法
Method hi = cls.getMethod("hi");
hi.setAccessible(true);//在反射调用方法时,取消访问检查
long start = System.currentTimeMillis();
for (int i = 0; i < 900000000; i++) {
hi.invoke(o);//反射机制调用方法
}
long end = System.currentTimeMillis();
System.out.println("m3()反射调用优化耗时=" + (end - start));
}
}

2.Class类
2.1基本介绍

Class类也是类,因此也继承Object类

Class类对象不是new出来的,而是系统创建的
对于某个类的Class类对象,在内存中只有一份,因为类只加载一次
每个类的实例都会记得自己是由哪个Class实例所生成
通过Class对象可以得到一个类的完整结构(通过一系列API)
Class对象是存放在堆的
类的字节码二进制数据,是放在方法区的,有的地方称为类的元数据(包括 方法代码,变量名,方法名,访问权限等)
当我们加载完类之后,除了会在堆里生成一个Class类对象,还会在方法区生成一个类的字节码二进制数据(元数据)
例子:
package li.reflection.class_;
import li.reflection.Cat;
//对Class类的特点的梳理
public class Class01 {
public static void main(String[] args) throws ClassNotFoundException {
//1.Class类对象不是new出来的,而是系统创建的
//1.1.传统的 new对象
/**通过ClassLoader类中的loadClass方法:
* public Class<?> loadClass(String name) throws ClassNotFoundException {
* return loadClass(name, false);
* }
*/
//Cat cat = new Cat();
//1.2反射的方式
/**在这里debug,需要先将上面的Cat cat = new Cat();注释掉,因为同一个类只加载一次,否则看不到loadClass方法
* (这里也验证了:3.对于某个类的Class类对象,在内存中只有一份,因为类只加载一次)
* 仍然是通过 ClassLoader类的loadClass方法加载 Cat类的 Class对象
* public Class<?> loadClass(String name) throws ClassNotFoundException {
* return loadClass(name, false);
* }
*/
Class cls1 = Class.forName("li.reflection.Cat");
//2.对于某个类的Class类对象,在内存中只有一份,因为类只加载一次
Class cls2 = Class.forName("li.reflection.Cat");
//这里输出的hashCode是相同的,说明cls1和cls2是同一个Class类对象
System.out.println(cls1.hashCode());//1554874502
System.out.println(cls2.hashCode());//1554874502
}
}
Class类对象不是new出来的,而是系统创建的:
- 在
Cat cat = new Cat();处打上断点,点击force step into,可以看到

- 注释
Cat cat = new Cat();,在Class cls1 = Class.forName("li.reflection.Cat");处打上断点,可以看到 仍然是通过 ClassLoader类加载 Cat类的 Class对

2.2Class类常用方法
public static Class<?> forName(String className)//传入完整的“包.类”名称实例化Class对象
public Constructor[] getContructors() //得到一个类的全部的构造方法
public Field[] getDeclaredFields()//得到本类中单独定义的全部属性
public Field[] getFields()//得到本类继承而来的全部属性
public Method[] getMethods()//得到一个类的全部方法
public Method getMethod(String name,Class..parameterType)//返回一个Method对象,并设置一个方法中的所有参数类型
public Class[] getInterfaces() //得到一个类中锁实现的全部接口
public String getName() //得到一个类完整的“包.类”名称
public Package getPackage() //得到一个类的包
public Class getSuperclass() //得到一个类的父类
public Object newInstance() //根据Class定义的类实例化对象
public Class<?> getComponentType() //返回表示数组类型的Class
public boolean isArray() //判断此class是否是一个数组
day42-反射01的更多相关文章
- 反射01 Class类的使用、动态加载类、类类型说明、获取类的信息
0 Java反射机制 反射(Reflection)是 Java 的高级特性之一,是框架实现的基础. 0.1 定义 Java 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对 ...
- Java反射01 : 概念、入门示例、用途及注意事项
1.Java反射定义 本文转载自:https://blog.csdn.net/hanchao5272/article/details/79360452 官方定义如下: Reflection enabl ...
- 03.反射--01【反射机制】【反射的应用场景】【Tomcat服务器】
https://blog.csdn.net/benjaminzhang666/article/details/9408611 https://blog.csdn.net/benjaminzhang66 ...
- 编程从入门到提高,然后放弃再跑路(Java)
1.Java入门篇 1.1 基础入门和面向对象 1.1.1 编程基础 [01] Java语言的基本认识 [02] 类和对象 [03] 类的结构和创建对象 [04] 包和访问权限修饰符 [05] 利用p ...
- 编程从入门到放弃(Java)
1.Java入门篇 1.1 基础入门和面向对象 1.1.1 编程基础 [01] Java语言的基本认识 [02] 类和对象 [03] 类的结构和创建对象 [04] 包和访问权限修饰符 [05] 利 ...
- java 反射机制01
// */ // ]]> java反射机制01 Table of Contents 1 反射机制 2 反射成员 2.1 java.lang.Class 2.2 Constructor 2.3 ...
- java反射机制入门01
java反射机制入门是我从极客学院的视频中学习的. 1.反射机制背景概述 反射(Reflection)是java被视为动态(或准动态)语言的一个关键性质.反射机制指的是程序在运行时能够获取任何类的内部 ...
- C#反射 入门学习 01
前言 获取方法的相关信息的两种形式 反射是一种允许用户获得类信息的C#功能,Type对象映射它代表的底层对象: 在.Net 中, 一旦获得了Type对象,就可以使用GetMethods()方法 ...
- java黑魔法-反射机制-01
在java的帮助文档中,java.lang包中有一个Class类,注意这里的"C“是大写,所以这个不是表示类的声明,而是一个真正的类.在java的帮助文档中,这样定义的Class类: pub ...
随机推荐
- V.Internet基础及应用
- PySide6/PyQt开发xml编辑器(1)
QTreeWidget折叠子项(折叠当前项的所有子项) 本文仅供本人知识总结使用,所以内容会比较浅显,不喜勿喷. 目录 QTreeWidget折叠子项(折叠当前项的所有子项) 目录 一.仅折叠子项 二 ...
- 聊聊 Redis 是如何进行请求处理
转载请声明出处哦~,本篇文章发布于luozhiyun的博客:https://www.luozhiyun.com/archives/674 本文使用的Redis 5.0源码 感觉这部分的代码还是挺有意思 ...
- angular变化检测OnPush策略需要注意的几个问题
OnPush组件内部触发的事件(包括viewChild)会引起组件的一次markForCheck Detached组件内部触发的事件不会引起组件的变化检测 OnPush组件的contentChild依 ...
- 转:windows下定时执行备份数据库
上一篇写了linux下定时任务,这一篇转发一个windows下定时备份数据库. 第一种:新建批处理文件 backup.dat,里面输入以下 net stop mysql xcopy "C:\ ...
- 【原创】Python 网易易盾滑块验证
本文仅供学习交流使用,如侵立删! 记一次 网易易盾滑块验证分析并通过 操作环境 win10 . mac Python3.9 selenium.PIL.numpy.scipy.matplotlib 分析 ...
- 万答17,AWS RDS怎么搭建本地同步库
欢迎来到 GreatSQL社区分享的MySQL技术文章,如有疑问或想学习的内容,可以在下方评论区留言,看到后会进行解答 背景说明 AWS RDS 权限受限,使用 mysqldump 的时候无法添加 - ...
- Apache SeaTunnel (Incubating) 2.1.0 发布,内核重构、全面支持 Flink
2021 年 12 月 9 日,SeaTunnel (原名 Waterdrop) 成功加入 Apache 孵化器,进入孵化器后,SeaTunnel 社区花费了大量时间来梳理整个项目的外部依赖以确保整个 ...
- Gulp介绍及安装使用教程
一.简介 gulp是前端开发过程中对代码进行构建的工具,是自动化项目的构建利器,不仅能对网站资源进行优化,而且在开发过程中很多重复的任务能够使用正确的工具自动完成,使用她,我们不仅可以很愉快的编写代码 ...
- identity4 系列————启航篇[二]
前言 开始identity的介绍了. 正文 前文介绍了一些概念,如果概念不清的话,可以去前文查看. https://www.cnblogs.com/aoximin/p/13475444.html 对一 ...