浅析Java反射--Java
前言
上篇文章我们提到了可以使用反射机制破解单例模式。这篇文章我们就来谈一谈什么是反射,反射有什么用,怎么用,怎么实现反射。
概述
Java的反射(reflection)机制:是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。这种动态获取程序信息以及动态调用对象的功能称为Java语言的反射机制。反射被视为动态语言的关键
功能
- 在运行时判断任意一个对象所属的类;
- 在运行时构造任意一个类的对象;
- 在运行时判断任意一个类所具有的成员变量和方法;
- 在运行时调用任意一个对象的方法;
- 生成动态代理。
java虽然不是动态语言,但是它却有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。
换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。
特点
优点:
- 能够运行时动态获取类的实例,大大提高系统的灵活性和扩展性。
- 与Java动态编译相结合,可以实现无比强大的功能
反射机制的功能非常强大,但不能滥用。在能不使用反射完成时,尽量不要使用,原因有以下几点:
缺点:
- 性能问题。
Java反射机制中包含了一些动态类型,所以Java虚拟机不能够对这些动态代码进行优化。因此,反射操作的效率要比正常操作效率低很多。我们应该避免在对性能要求很高的程序或经常被执行的代码中使用反射。而且,如何使用反射决定了性能的高低。如果它作为程序中较少运行的部分,性能将不会成为一个问题。 - 安全限制。
使用反射通常需要程序的运行没有安全方面的限制。如果一个程序对安全性提出要求,则最好不要使用反射。 - 程序健壮性。
反射允许代码执行一些通常不被允许的操作,所以使用反射有可能会导致意想不到的后果。反射代码破坏了Java程序结构的抽象性,所以当程序运行的平台发生变化的时候,由于抽象的逻辑结构不能被识别,代码产生的效果与之前会产生差异。
获取Class对象的三种方式
三种方式
- Class.forName("全类名") 多用于配置文件时
- 类名.class 多用于参数传递时
- 对象.getClass 多用于有对象实例时
==我们创建一个Person类用于实验,分别使用三种方法获取他们的对象,并打印他们的hashCode(),我们会发现他们的hashcode是相同的,证明他们是获取的是相同的,所有统一个字节码文件(*.class)在一次运行中,只会被加载一次。不论通过哪一种方式获取的对象都是同一个。
package hello;
public class ReflectDemo {
public static void main(String[] args) throws Exception{
//Class.forName("全类名")
Class<?> aClass = Class.forName("hello.Person");
System.out.println(aClass);
System.out.println(aClass.hashCode());
//类名.clss
Class<Person> personClass = Person.class;
System.out.println(personClass);
System.out.println(personClass.hashCode());
//对象.getClass()
Person person = new Person();
System.out.println(person.getClass());
System.out.println(person.getClass().hashCode());
}
}
class Person{
private String name;
public Person(){
}
public Person(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Class对象功能
Class对象里面大概有120多个方法,我们不可能全部都记住,也没必要全部都记住,我们只需要记住常用的方法和功能就可以了。
获取功能
获取成员变量
- Field[] getFields()
- Field getFields(String name)
- FIeld[] getDeclaredFields()
- FIeld getDeclaredField(String name)
获取构造方法
- Constructor<?>[] getConstructors()
- Constructor getConstructor(类<?>..., parameterTypes)
- Constructor getDeclaredConstructor(类<?>..., parameterTypes)
- Constructor<?>[] getDeclaredConstructors()
获取成员方法
- Method[] getMethods()
- Method getMethod(String name,类<?>..., parameterType)
- Method[] getDeclaredMethods()
- Method getDeclaredMethod(String name,类<?>...,parameterTypes)
获取类名
- String getName()
| 方法 | 作用 |
|---|---|
| Field[] getFields() | 获取public的成员变量名 |
| Field getFields(String name) | 获取public的成员变量名,需要指定名称 |
| FIeld[] getDeclaredFields() | 获取所有的成员变量,包括private |
| FIeld getDeclaredField(String name) | 获取指定的成员变量,私有的也可以获取,如果要改这个变量的值需要setAccessible(ture)暴力反射 |
| Constructor<?>[] getConstructors() | 用于返回一个构造函数对象数组,该数组反映此Class对象表示的类的所有公共构造函数。 |
| Constructor getConstructor(类<?>..., parameterTypes) | 获取指定参数的构造方法 |
| Constructor getDeclaredConstructor(类<?>..., parameterTypes) | 方法返回一个 Constructor 对象,该对象反映了此 Class 对象表示的类或接口的指定构造函数 |
| Constructor<?>[] getDeclaredConstructors() | 用于返回一个Constructor对象数组,该数组指示此Class对象所表示的类定义的构造函数的类型(Constructor可以是public,private,protected或default) |
| Method[] getMethods() | 公共的所有方法 |
| Method getMethod(String name,类<?>..., parameterType) | 指定公共的其中的方法,parameterType时一个数组 |
| Method[] getDeclaredMethods() | 所有的包括私有的 |
| Method getDeclaredMethod(String name,类<?>...,parameterTypes) | 指定的,parameterTypes表示数组 |
| String getName() | 获取类名程 |
package hello;
import com.sun.org.apache.xerces.internal.impl.XMLEntityScanner;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class ReflectDemo {
public static void main(String[] args) throws Exception{
//获取class对象
Class<Person> personClass = Person.class;
//Field[] getFields()
Field[] fields = personClass.getFields();
for (Field field : fields) {
System.out.println(field);
}
System.out.println("******************************");
//FIeld[] getDeclaredFields()
Field[] declaredFields = personClass.getDeclaredFields();
for (Field declaredField : declaredFields) {
System.out.println(declaredField);
}
System.out.println("******************************");
//Constructor<?>[] getConstructors()
Constructor<Person> constructor = personClass.getConstructor();
System.out.println(constructor);
System.out.println("******************************");
//Method[] getMethods()
Method[] methods = personClass.getMethods();
for (Method method : methods) {
System.out.println(method);
}
System.out.println("******************************");
//String getName()
System.out.println(personClass.getName());
}
}
class Person{
public int id;
private String name;
public Person(){
}
public Person(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
案例
创建一个功能,不改里面的代码,只需要修改配置文件就能调用不同的方法
第一步:创建配置文件,”data.properties"
这个路径一定要写成hello.Preson,写成斜杠会识别不了。
# 这个写全路径类名
className=hello.Preson
# 这个写类里面想要调用的方法
methodName=eat
第二步:具体类"Preson.java"
package hello;
public class Preson {
public String name;
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void eat(){
System.out.println("eat....");
}
}
第三步:编写实现代码,名成ReflectDemo.java
package hello;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.Properties;
public class ReflectDemo {
public static void main(String[] args) throws Exception{
//加载配置文件
//创建properties对象
Properties properties = new Properties();
//加载配置文件,转为一个集合
ClassLoader classLoader = ReflectDemo.class.getClassLoader();
InputStream resourceAsStream = classLoader.getResourceAsStream("hello/data.properties");
properties.load(resourceAsStream);
//获取配置文件定义的数据
String className = properties.getProperty("className");
String methodName = properties.getProperty("methodName");
//加载该类进入内存
Class aClass = Class.forName(className);
Object o = aClass.newInstance();
Method method = aClass.getMethod(methodName);
method.invoke(o);
}
}
第四步:运行,我们只需要修改properties文件里面的类名和方法就能实现不同类不同方法的创建了。
浅析Java反射--Java的更多相关文章
- java反射 java动态代理和cglib动态代理的区别
java反射 https://blog.csdn.net/f2764052703/article/details/89311013 java 动态代理 https://blog.csdn ...
- java反射+java泛型,封装BaseDaoUtil类。供应多个不同Dao使用
当项目是ssh框架时,每一个Action会对应一个Service和一个Dao.但是所有的Ation对应的Dao中的方法是相同的,只是要查的表不一样.由于封装的思想,为了提高代码的重用性.可以使用jav ...
- 解析Java反射java.lang.IllegalArgumentException: wrong number of arguments
项目中遇到的问题 import org.springframework.util.ReflectionUtils; import java.lang.reflect.Method; public cl ...
- Java反射——java.lang.Class和类的加载
反射的基础: java.lang.Class Class类的实例对象,用于记录类描述信息. 源码说:represent classes and interfaces in a running Java ...
- Java反射——java.lang.Class 类简介
Java的基本思想之一是万事万物即对象,类也是一种对象.但是类是什么对象呢?Java中的类是java.lang.Class的实例化对象,这被成为类类型. //java.lang.Class类中的的主要 ...
- java反射知识
java反射机制是在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称 ...
- java反射机制简单实例
目录 Java反射 简单实例 @(目录) Java反射 Java语言允许通过程序化的方式间接对Class进行操作.Class文件由类装载器装载后,在JVM中将形成一份描述Class结构的元信息对象,通 ...
- Java反射及其在Android中的应用学习总结
一. Java反射机制 Reflection 是Java被视为动态(或准动态)语言的一个关键性质.这个机制同意程序在执行时透过Reflection APIs取得不论什么一个已知名称的class的内部信 ...
- java反射(1)
反射反射,程序员的快乐. 第一次了解反射这个概念是从<<大话设计>>中所了解到的.当时只是对概念的模糊了解,具体对它的机制并不清楚.最近在学习并实践SSH框架,其中Spring ...
随机推荐
- time模块以及datetime模块
内容概要 time模块 **timestamp时间戳 **struct_time结构化时间 **format time格式化时间 datetime模块 **date **time **datetime ...
- python生成器对象&常见内置函数
内容概要 异常捕获(补充) for循环本质 生成器 yield 和 return优缺点 笔试题 常用内置函数 内容详细 一.异常捕获补充 try: print(name) except NameErr ...
- Note/Solution - 转置原理 & 多点求值
\[\newcommand{\vct}[1]{\boldsymbol{#1}} \newcommand{\mat}[1]{\begin{bmatrix}#1\end{bmatrix}} \newcom ...
- 最新版的Dubbo Admin 3.0 本地启动方式
项目下载 项目地址:https://github.com/apache/dubbo-admin 如下图,使用git地址直接构建或者下载zip包构建源码都可以,我用的是下载的zip包, 项目架构说明 d ...
- Ultra Math Preview : VSCode上的LaTeX公式实时预览插件
一直觉着 VS Code 的 TeX 公式(包括 markdown 和 LaTeX)只能在定界符上hover预览,或者开一个预览面板看,没有那种像 Typora 一样紧跟在公式后面的预览面板,多少有些 ...
- spring IOC的理解,原理与底层实现?
从总体到局部 总 控制反转:理论思想,原来的对象是由使用者来进行控制,有了spring之后,可以把整个对象交给spring来帮我们进行管理 DI(依赖注入):把对应的属性 ...
- IDEA如何快速生成get和set方法
方法一:1.鼠标右击"Generate"2.点击"Getter and Setter",3.将定义的字段全部选中,点击OK.方法二:使用alt+insert 快 ...
- 利用AWVS扫描Web漏洞
实验目的 利用AWVS扫描Web漏洞. 实验原理 AWVS是一款知名的网络漏洞扫描工具,它通过网络爬虫测试你的网站安全,检测流行安全漏洞. 实验内容 AWVS是一个自动化的web应用程序安全测试工具, ...
- 企业没有大数据技术?选择这款BI工具
无论是网络时代的传统营销还是大数据营销,营销人员的任务之一就是找到目标客户,实现自己的营销目标.而我们说的大数据营销只不过是营销的工具发生了变化,营销的本质和目标是不变的. 就目前而言,现在的大数据 ...
- 【CPU】进程管理之五状态模型
本文为第三篇,进程管理之五状态模型,进程在操作系统里边是有多个状态的,本文就是了解进程在操作系统中的多个状态 1.进程的五个状态 创建状态 就绪状态 阻塞状态 执行状态 终止状态 2.进程处于这五种状 ...