187 lines
5.9 KiB
Markdown
187 lines
5.9 KiB
Markdown
# 反射
|
||
## Class对象
|
||
Class对象用于访问类信息,想要获取class对象,可以通过如下方法
|
||
### 获取Class对象方法
|
||
#### 调用实例的getClass方法
|
||
```java
|
||
Random generator = new Random():
|
||
Class cl = generator . getClass () ;
|
||
String name = cl.getName() ; // name is set to "java . util . Random"
|
||
```
|
||
#### 调用Class类的静态方法forName方法
|
||
```java
|
||
String dassName = " java . util . Random" ;
|
||
Class cl = Class.forName ( dassName ) ;
|
||
```
|
||
|
||
#### 通过T.class获取
|
||
如果T是一个java类型,那么T.class代表T类型对应的Class对象
|
||
```java
|
||
Class dl = Random.class;
|
||
Gass cl 2 = int.class ;
|
||
Class cl 3 = Double[].class ;
|
||
```
|
||
Class对象表示的是一个类型,而类型不一定是类,也包含基本类型,如int,long等。
|
||
|
||
### Class对象的比较
|
||
JVM为每个类型只维护一个Class对象,故而不同的Class之间可以通过`==`符号进行比较
|
||
```java
|
||
if(e.getClass()==Employee.class)
|
||
```
|
||
### 通过Class对象创建新实例
|
||
下述示例创建了一个与e具有相同类型的实例,newInstance方法默认会调用默认构造函数,如果该类型没有默认构造函数,会抛出一个checked exception
|
||
```java
|
||
e.getClass().newInstance()
|
||
```
|
||
如果将forName和newInstance方法结合起来使用,可以根据存储在字符串中的类型动态创建一个对象:
|
||
```java
|
||
String driverClassName = "com.mysql.cj.jdbc.Driver";
|
||
Class.forName(driverClassName).newInstance();
|
||
```
|
||
|
||
## 使用反射分析类
|
||
java.lang.reflect类中含有三个类:
|
||
- Field:用于描述类的域
|
||
- Method:用于描述类的方法
|
||
- Constructor:用于描述类的构造器
|
||
|
||
### Field.getType
|
||
Field类有一个getType方法,返回值是域所属类型的Class对象。
|
||
### 获取类的域、方法和构造器
|
||
#### Class.getFields
|
||
返回值为类的public域,包含从父类继承的public域
|
||
#### Class.getDeclaredFields
|
||
返回类中声明的所有域,包含为private和protected的域,但是不包含从父类继承的域
|
||
#### Class.getMethods
|
||
返回值为类提供的public方法,包含从父类继承的public方法
|
||
#### Class.getDeclaredMethods
|
||
返回类中声明的所有方法,包含private方法和protected方法,但是不包含从父类继承的方法
|
||
#### Class.getConstructors
|
||
只能获取到public构造器
|
||
#### Class.getDeclaredConstructors
|
||
能获取到所有的构造器,包含private和protected构造器
|
||
### Method、Field、Constructor中的方法
|
||
#### getDeclaringClass
|
||
返回定义该方法、域、构造器类型的Class对象
|
||
```java
|
||
Class getDeclaringClass()
|
||
```
|
||
#### getExceptionTypes(Method和Constructor中包含)
|
||
返回方法抛出的异常类型的数组
|
||
```java
|
||
Class[] getExceptionTypes()
|
||
```
|
||
#### getModifiers
|
||
返回一个描述Field、Method、Constrcutor的修饰符的整数类型(不同的位描述不同的修饰符状态,如public、static、是否抽象类、是否接口等)
|
||
```java
|
||
int getModifiers()
|
||
```
|
||
#### getName
|
||
返回描述Field、Constructor、Method的字符串
|
||
```java
|
||
String getName()
|
||
```
|
||
#### getParameterTypes(Method和Constructor中包含)
|
||
返回一个Class数组,用于描述参数类型
|
||
```java
|
||
Class[] getParameterTypes()
|
||
```
|
||
#### getReturnType(只在Method方法中包含)
|
||
返回Method的返回类型
|
||
```java
|
||
Class getReturnType()
|
||
```
|
||
### Modifier中的方法
|
||
#### toString
|
||
该静态方法将modifiers整数中的修饰符内容转化为字符串
|
||
```java
|
||
static String toString(int modifiers)
|
||
```
|
||
#### isAbstract
|
||
```java
|
||
static boolean isAbstract(int modifiers)
|
||
```
|
||
#### isFinal
|
||
```java
|
||
static boolean isFinal(int modifiers)
|
||
```
|
||
#### isInterface
|
||
```java
|
||
static boolean isInterface(int modifiers)
|
||
```
|
||
#### isNative
|
||
```java
|
||
static boolean isNative(int modifiers)
|
||
```
|
||
#### isPrivate
|
||
```java
|
||
static boolean isPrivate(int modifiers)
|
||
```
|
||
#### isProtected
|
||
```java
|
||
static boolean isProtected(int modifiers)
|
||
```
|
||
#### isPublic
|
||
```java
|
||
static boolean isPublic(int modifiers)
|
||
```
|
||
#### isStatic
|
||
```java
|
||
static boolean isStatic(int modifiers)
|
||
```
|
||
#### isSynchronized
|
||
```java
|
||
static boolean isSynchronized(int modifiers)
|
||
```
|
||
#### isVolatile
|
||
```java
|
||
static boolean isVolatile(int modifiers)
|
||
```
|
||
## 使用反射分析类的内容
|
||
如果在运行时有一个Field对象和一个类实例obj,想要查看obj实例中Field对应域的值,可以使用如下方式:
|
||
```java
|
||
f.get(obj);
|
||
```
|
||
但是,如果Field的modifiers显示该Field是private的,那么`f.get(obj)`方法将会抛出一个IllegalAccessException。
|
||
此时,可以通过`f.setAccessible(true)`来访问private域中的内容。
|
||
```java
|
||
f.setAccessible(true);
|
||
Object v = f.get(obj);
|
||
```
|
||
设置Field的值则是可以通过`f.set(obj,value)`来实现。
|
||
```java
|
||
f.setAccessible(true);
|
||
f.setValue(obj,"shiro");
|
||
```
|
||
### AccessibleObject
|
||
Field,Method,Constructor类型都拥有公共的父类AccessibleObject
|
||
#### setAccessible
|
||
```java
|
||
void setAccessible(boolean flag);
|
||
|
||
// 批量设置可访问状态
|
||
static void setAccessible(AccessibleObject[] array,boolean flag);
|
||
```
|
||
#### isAccessible
|
||
```java
|
||
boolean isAccessible();
|
||
```
|
||
## 使用反射调用方法
|
||
Method类中提供了一个invoke方法来调用Method所表示的方法,`Method.invoke`方法签名如下:
|
||
```java
|
||
Object invoke(Object obj,Object... args)
|
||
```
|
||
obj表示调用方法的对象,args表示参数数组。
|
||
> 如果Method位静态方法,那么obj应该传入null
|
||
|
||
如果Method代表的方法返回的是一个基本类型,那么invoke方法会返回器包装类型,如`int->Integer`.
|
||
### Method.getMethod
|
||
如果想要根据方法名称和参数类型来获取Class对象中的Method,可以调用如下方法:
|
||
```java
|
||
Method getMethod(String name,Class... parameterTypes)
|
||
```
|
||
示例如下:
|
||
```java
|
||
Method ml = Employee. class.getMethod ( "getName") ;
|
||
Method m2 = Employee.class.getMethod ( "raiseSalary" , double.class) ;
|
||
``` |