mirror of
https://github.com/Snailclimb/JavaGuide
synced 2025-06-16 18:10:13 +08:00
反射单独成文
This commit is contained in:
parent
2aed7e357e
commit
172a11c54b
@ -92,6 +92,7 @@ Github用户如果访问速度缓慢的话,可以转移到[码云](https://git
|
||||
|
||||
1. [枚举](docs/java/basis/用好Java中的枚举真的没有那么简单.md) (很重要的一个数据结构,用好枚举真的没有那么简单!)
|
||||
2. [Java 常见关键字总结:final、static、this、super!](docs/java/basis/final、static、this、super.md)
|
||||
3. [什么是反射机制?反射机制的应用场景有哪些?](docs/java/basis/reflection.md)
|
||||
|
||||
### 容器
|
||||
|
||||
|
135
docs/java/Basis/reflection.md
Normal file
135
docs/java/Basis/reflection.md
Normal file
@ -0,0 +1,135 @@
|
||||
### 反射机制介绍
|
||||
|
||||
JAVA 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 java 语言的反射机制。
|
||||
|
||||
### 获取 Class 对象的两种方式
|
||||
|
||||
如果我们动态获取到这些信息,我们需要依靠 Class 对象。Class 类对象将一个类的方法、变量等信息告诉运行的程序。Java 提供了两种方式获取 Class 对象:
|
||||
|
||||
1.知道具体类的情况下可以使用:
|
||||
|
||||
```java
|
||||
Class alunbarClass = TargetObject.class;
|
||||
```
|
||||
|
||||
但是我们一般是不知道具体类的,基本都是通过遍历包下面的类来获取 Class 对象
|
||||
|
||||
2.通过 `Class.forName()`传入类的路径获取:
|
||||
|
||||
```java
|
||||
Class alunbarClass1 = Class.forName("cn.javaguide.TargetObject");
|
||||
```
|
||||
|
||||
### 代码实例
|
||||
|
||||
**简单用代码演示一下反射的一些操作!**
|
||||
|
||||
1.创建一个我们要使用反射操作的类 `TargetObject`:
|
||||
|
||||
```java
|
||||
package cn.javaguide;
|
||||
|
||||
public class TargetObject {
|
||||
private String value;
|
||||
|
||||
public TargetObject() {
|
||||
value = "JavaGuide";
|
||||
}
|
||||
|
||||
public void publicMethod(String s) {
|
||||
System.out.println("I love " + s);
|
||||
}
|
||||
|
||||
private void privateMethod() {
|
||||
System.out.println("value is " + value);
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
2.使用反射操作这个类的方法以及参数
|
||||
|
||||
```java
|
||||
package cn.javaguide;
|
||||
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.InvocationTargetException;
|
||||
import java.lang.reflect.Method;
|
||||
|
||||
public class Main {
|
||||
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException, NoSuchFieldException {
|
||||
/**
|
||||
* 获取TargetObject类的Class对象并且创建TargetObject类实例
|
||||
*/
|
||||
Class<?> tagetClass = Class.forName("cn.javaguide.TargetObject");
|
||||
TargetObject targetObject = (TargetObject) tagetClass.newInstance();
|
||||
/**
|
||||
* 获取所有类中所有定义的方法
|
||||
*/
|
||||
Method[] methods = tagetClass.getDeclaredMethods();
|
||||
for (Method method : methods) {
|
||||
System.out.println(method.getName());
|
||||
}
|
||||
/**
|
||||
* 获取指定方法并调用
|
||||
*/
|
||||
Method publicMethod = tagetClass.getDeclaredMethod("publicMethod",
|
||||
String.class);
|
||||
|
||||
publicMethod.invoke(targetObject, "JavaGuide");
|
||||
/**
|
||||
* 获取指定参数并对参数进行修改
|
||||
*/
|
||||
Field field = tagetClass.getDeclaredField("value");
|
||||
//为了对类中的参数进行修改我们取消安全检查
|
||||
field.setAccessible(true);
|
||||
field.set(targetObject, "JavaGuide");
|
||||
/**
|
||||
* 调用 private 方法
|
||||
*/
|
||||
Method privateMethod = tagetClass.getDeclaredMethod("privateMethod");
|
||||
//为了调用private方法我们取消安全检查
|
||||
privateMethod.setAccessible(true);
|
||||
privateMethod.invoke(targetObject);
|
||||
}
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
输出内容:
|
||||
|
||||
```
|
||||
publicMethod
|
||||
privateMethod
|
||||
I love JavaGuide
|
||||
value is JavaGuide
|
||||
```
|
||||
|
||||
### 静态编译和动态编译
|
||||
|
||||
- **静态编译:**在编译时确定类型,绑定对象
|
||||
- **动态编译:**运行时确定类型,绑定对象
|
||||
|
||||
### 反射机制优缺点
|
||||
|
||||
- **优点:** 运行期类型的判断,动态加载类,提高代码灵活度。
|
||||
- **缺点:** 1,性能瓶颈:反射相当于一系列解释操作,通知 JVM 要做的事情,性能比直接的 java 代码要慢很多。2,安全问题,让我们可以动态操作改变类的属性同时也增加了类的安全隐患。
|
||||
|
||||
### 反射的应用场景
|
||||
|
||||
**反射是框架设计的灵魂。**
|
||||
|
||||
在我们平时的项目开发过程中,基本上很少会直接使用到反射机制,但这不能说明反射机制没有用,实际上有很多设计、开发都与反射机制有关,例如模块化的开发,通过反射去调用对应的字节码;动态代理设计模式也采用了反射机制,还有我们日常使用的 Spring/Hibernate 等框架也大量使用到了反射机制。
|
||||
|
||||
举例:
|
||||
|
||||
1. 我们在使用 JDBC 连接数据库时使用 `Class.forName()`通过反射加载数据库的驱动程序;
|
||||
2. Spring 框架的 IOC(动态加载管理 Bean)创建对象以及 AOP(动态代理)功能都和反射有联系;
|
||||
3. 动态配置实例的属性;
|
||||
4. ......
|
||||
|
||||
**推荐阅读:**
|
||||
|
||||
- [Reflection:Java 反射机制的应用场景](https://segmentfault.com/a/1190000010162647?utm_source=tuicool&utm_medium=referral)
|
||||
- [Java 基础之—反射(非常重要)](https://blog.csdn.net/sinat_38259539/article/details/71799078)
|
||||
|
||||
##
|
@ -55,14 +55,7 @@
|
||||
- [BIO,NIO,AIO 有什么区别?](#bionioaio-有什么区别)
|
||||
- [36. 常见关键字总结:static,final,this,super](#36-常见关键字总结staticfinalthissuper)
|
||||
- [37. Collections 工具类和 Arrays 工具类常见方法总结](#37-collections-工具类和-arrays-工具类常见方法总结)
|
||||
- [38. 深拷贝 vs 浅拷贝](#38-深拷贝-vs-浅拷贝)
|
||||
- [38.什么是反射机制?反射机制的应用场景有哪些?](#38什么是反射机制反射机制的应用场景有哪些)
|
||||
- [反射机制介绍](#反射机制介绍)
|
||||
- [获取 Class 对象的两种方式](#获取-class-对象的两种方式)
|
||||
- [代码实例](#代码实例)
|
||||
- [静态编译和动态编译](#静态编译和动态编译)
|
||||
- [反射机制优缺点](#反射机制优缺点)
|
||||
- [反射的应用场景](#反射的应用场景)
|
||||
- [38.深拷贝 vs 浅拷贝](#38-深拷贝-vs-浅拷贝)
|
||||
- [参考](#参考)
|
||||
- [公众号](#公众号)
|
||||
|
||||
@ -548,149 +541,13 @@ Java Io 流共涉及 40 多个类,这些类看上去很杂乱,但实际上
|
||||
|
||||
详见笔主的这篇文章: <https://gitee.com/SnailClimb/JavaGuide/blob/master/docs/java/Basis/Arrays,CollectionsCommonMethods.md>
|
||||
|
||||
### 38. 深拷贝 vs 浅拷贝
|
||||
## 38. 深拷贝 vs 浅拷贝
|
||||
|
||||
1. **浅拷贝**:对基本数据类型进行值传递,对引用数据类型进行引用传递般的拷贝,此为浅拷贝。
|
||||
2. **深拷贝**:对基本数据类型进行值传递,对引用数据类型,创建一个新的对象,并复制其内容,此为深拷贝。
|
||||
|
||||

|
||||
|
||||
## 38.什么是反射机制?反射机制的应用场景有哪些?
|
||||
|
||||
### 反射机制介绍
|
||||
|
||||
JAVA 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 java 语言的反射机制。
|
||||
|
||||
### 获取 Class 对象的两种方式
|
||||
|
||||
如果我们动态获取到这些信息,我们需要依靠 Class 对象。Class 类对象将一个类的方法、变量等信息告诉运行的程序。Java 提供了两种方式获取 Class 对象:
|
||||
|
||||
1.知道具体类的情况下可以使用:
|
||||
|
||||
```java
|
||||
Class alunbarClass = TargetObject.class;
|
||||
```
|
||||
|
||||
但是我们一般是不知道具体类的,基本都是通过遍历包下面的类来获取 Class 对象
|
||||
|
||||
2.通过 `Class.forName()`传入类的路径获取:
|
||||
|
||||
```java
|
||||
Class alunbarClass1 = Class.forName("cn.javaguide.TargetObject");
|
||||
```
|
||||
|
||||
### 代码实例
|
||||
|
||||
**简单用代码演示一下反射的一些操作!**
|
||||
|
||||
1.创建一个我们要使用反射操作的类 `TargetObject`:
|
||||
|
||||
```java
|
||||
package cn.javaguide;
|
||||
|
||||
public class TargetObject {
|
||||
private String value;
|
||||
|
||||
public TargetObject() {
|
||||
value = "JavaGuide";
|
||||
}
|
||||
|
||||
public void publicMethod(String s) {
|
||||
System.out.println("I love " + s);
|
||||
}
|
||||
|
||||
private void privateMethod() {
|
||||
System.out.println("value is " + value);
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
2.使用反射操作这个类的方法以及参数
|
||||
|
||||
```java
|
||||
package cn.javaguide;
|
||||
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.InvocationTargetException;
|
||||
import java.lang.reflect.Method;
|
||||
|
||||
public class Main {
|
||||
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException, NoSuchFieldException {
|
||||
/**
|
||||
* 获取TargetObject类的Class对象并且创建TargetObject类实例
|
||||
*/
|
||||
Class<?> tagetClass = Class.forName("cn.javaguide.TargetObject");
|
||||
TargetObject targetObject = (TargetObject) tagetClass.newInstance();
|
||||
/**
|
||||
* 获取所有类中所有定义的方法
|
||||
*/
|
||||
Method[] methods = tagetClass.getDeclaredMethods();
|
||||
for (Method method : methods) {
|
||||
System.out.println(method.getName());
|
||||
}
|
||||
/**
|
||||
* 获取指定方法并调用
|
||||
*/
|
||||
Method publicMethod = tagetClass.getDeclaredMethod("publicMethod",
|
||||
String.class);
|
||||
|
||||
publicMethod.invoke(targetObject, "JavaGuide");
|
||||
/**
|
||||
* 获取指定参数并对参数进行修改
|
||||
*/
|
||||
Field field = tagetClass.getDeclaredField("value");
|
||||
//为了对类中的参数进行修改我们取消安全检查
|
||||
field.setAccessible(true);
|
||||
field.set(targetObject, "JavaGuide");
|
||||
/**
|
||||
* 调用 private 方法
|
||||
*/
|
||||
Method privateMethod = tagetClass.getDeclaredMethod("privateMethod");
|
||||
//为了调用private方法我们取消安全检查
|
||||
privateMethod.setAccessible(true);
|
||||
privateMethod.invoke(targetObject);
|
||||
}
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
输出内容:
|
||||
|
||||
```
|
||||
publicMethod
|
||||
privateMethod
|
||||
I love JavaGuide
|
||||
value is JavaGuide
|
||||
```
|
||||
|
||||
### 静态编译和动态编译
|
||||
|
||||
- **静态编译:**在编译时确定类型,绑定对象
|
||||
- **动态编译:**运行时确定类型,绑定对象
|
||||
|
||||
### 反射机制优缺点
|
||||
|
||||
- **优点:** 运行期类型的判断,动态加载类,提高代码灵活度。
|
||||
- **缺点:** 1,性能瓶颈:反射相当于一系列解释操作,通知 JVM 要做的事情,性能比直接的 java 代码要慢很多。2,安全问题,让我们可以动态操作改变类的属性同时也增加了类的安全隐患。
|
||||
|
||||
### 反射的应用场景
|
||||
|
||||
**反射是框架设计的灵魂。**
|
||||
|
||||
在我们平时的项目开发过程中,基本上很少会直接使用到反射机制,但这不能说明反射机制没有用,实际上有很多设计、开发都与反射机制有关,例如模块化的开发,通过反射去调用对应的字节码;动态代理设计模式也采用了反射机制,还有我们日常使用的 Spring/Hibernate 等框架也大量使用到了反射机制。
|
||||
|
||||
举例:
|
||||
|
||||
1. 我们在使用 JDBC 连接数据库时使用 `Class.forName()`通过反射加载数据库的驱动程序;
|
||||
2. Spring 框架的 IOC(动态加载管理 Bean)创建对象以及 AOP(动态代理)功能都和反射有联系;
|
||||
3. 动态配置实例的属性;
|
||||
4. ......
|
||||
|
||||
**推荐阅读:**
|
||||
|
||||
- [Reflection:Java 反射机制的应用场景](https://segmentfault.com/a/1190000010162647?utm_source=tuicool&utm_medium=referral)
|
||||
- [Java 基础之—反射(非常重要)](https://blog.csdn.net/sinat_38259539/article/details/71799078)
|
||||
|
||||
## 参考
|
||||
|
||||
- https://stackoverflow.com/questions/1906445/what-is-the-difference-between-jdk-and-jre
|
||||
|
Loading…
x
Reference in New Issue
Block a user