Java反射详解

简介:

对象的编译类型和运行类型


对象有编译类型和运行类型
1
Object obj = new Date();


编译类型:Object
运行类型(其实就是obj对象真实的类型):Date
需求:根据对象obj调用Date类中的一个方法,toLocaleString,如何来做?
obj.toLocaleString()代码在编译阶段去编译类型Object中检查是否有该方法,若没有,编译失败。

解决方案:强制转为obj为Date类型,前提:必须知道对象的真实类型是什么?
1
2
Date d = (Date)obj;
d.toLocaleString();


万物皆对象:从概念分析类也是一种对象,新的问题,既然类是一种对象,那么谁又来描述该对象呢?而我们又说描述对象的类都是对象
元数据:metadata描述数据的描述数据
反射:得到元数据的行为,既然Class是对一切类共同功能行为和状态的抽象
那么Class重有应该有类具备的成员:
getPackage() 表示类的包
getSuperClass() 表示类的父类
getMethods() 类的方法
getField() 类的字段

Class类和Class实例


Class类:用于描述一切类/接口,枚举是一种类,注解是一种借口。
Class实例:就是指JVM中的一份字节码
问题:那Class实例到底表示的是哪一份字节码,为了明确区分出Class实例表示的是谁的字节码。Class类提供了泛型。
Class clz1 = Date.class // clz1表示的是Date的字节码
Class clz2 = String.class // clz2表示的是String的字节码

如何得到Class的实例?
1、类型.class(就是一份字节码)
2、Class.forName(String className);根据一个类的全限定名来构建Class对象
3、每一个对象都有getClass()方法,obj.getClss();返回对象的真实类型

9个预定义Class对象


基本的Java类型(boolean、byte、char、short、int、long、float和double)和关键字void也表示为Class对象
表示int的Class对象:Class clz = int.class;
表示boolean的Class对象:boolean.class;
表示void的Class对象:void.class;

所有的数据类型都有class属性,表示都是Class对象。
在八大基本数据类型的包装类中都有一个常量:TYPE
TYPE表示的是该包装类对应的基本数据类型的Class实例
如Integer.TYPE == int.class //true
Integer.class == int.class //false

数组的Class实例


String[] sArr = {“A”, “C”};
String[] sArr2 = {};
String[][] sArr3 = {};
int[] sArr4 = {};
表示数组的Class实例:
所有具有相同元素类型和维数的数组都共享该Class对象。
注意:和数组中的元素没有一点关系。

反射的实例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
package 面试题;

import java.util.Arrays;

/**
* Created by piqiu on 3/3/16.
*/

public class User {

private String name;
private int age;
private int sex;

public User() {
System.out.println("public User()");
}

private User(String name) {
this.name = name;
System.out.println("private User(String name): " + name);
}

public User(String name, int age, int sex) {
this.name = name;
this.age = age;
this.sex = sex;
System.out.println("public User(String name, int age, int sex): " + name + " " + age + " " + sex);
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public int getSex() {
return sex;
}

public void setSex(int sex) {
this.sex = sex;
}

private void sayHello() {
System.out.println("hello...");
}

public void sayHi(String name) {
System.out.println("hi..." + name);
}

public double getMoney(String name) {
double money = 200.12;
System.out.println("get money " + money + " for " + name);
return money;
}

public static void staticMethod() {
System.out.println("invoke static method...");
}

public static int show(int... args) {
int ret = 0;
for (int i : args) {
ret += i;
}
return ret;
}

public static void show2(String... args) {
System.out.println(Arrays.toString(args));
}

@Override
public String toString() {

return "User -> name: " + ((name == null)?"无名氏":name) + " age: " +
((age == 0)?"无年龄":age) + " sex: " + ((sex == 1)?"男":(sex == 2)?"女":"无性别");
}
}
1
2
3
4
5
6
7
8
9
10
11
package 面试题;

import java.util.Map;

/**
* Created by piqiu on 3/7/16.
*/

public class OOXX {

public Map<String, Object> cache;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
package 面试题;

import java.lang.reflect.*;
import java.util.Arrays;
import java.util.List;

/**
* Created by piqiu on 3/3/16.
*/
public class ReflectionDemo {

private static Class<User> clz = User.class;
private static Class<OOXX> ooxxClass = OOXX.class;

private ReflectionDemo(){}

public static void main(String[] args) throws Exception {
// getConstructors();
// getConstructor();
// getDeclaredConstructors();
// getDeclaredConstructor();
// invokeConstructor();
// getAllMethods();
// getOneMethod();
// invokeMethod();
// invokeStaticMethod();
// invokeVariableLengthMethod();
// getAllFields();
// invokeField();
// getFieldValue();
// getOtherAPI();
// invokeGenericMethod();
getGenericFieldType();
}

/**
* 获取所有的【public修饰】的构造器
*/
private static void getConstructors() {
Constructor<?>[] constructors = clz.getConstructors();
for (Constructor c : constructors) {
System.out.println(c);
System.out.println(Arrays.toString(c.getParameterTypes()));
System.out.println(c.getDeclaringClass());
}
/**
public 面试题.User(java.lang.String,int,int)
[class java.lang.String, int, int]
class 面试题.User
public 面试题.User()
[]
class 面试题.User
*/
}

/**
* 根据构造器参数的Class类型获得对应的Constructor,只能获取【public修饰】的
* @throws NoSuchMethodException
*/
private static void getConstructor() throws NoSuchMethodException {
Constructor<User> constructor1 = clz.getConstructor();
System.out.println(constructor1); // public 面试题.User()
Constructor<User> constructor3 = clz.getConstructor(String.class, int.class, int.class);
System.out.println(constructor3); // public 面试题.User(java.lang.String,int,int)
}

/**
* 获取所有构造器,和访问权限无关
*/
private static void getDeclaredConstructors() {
Constructor<?>[] declaredConstructors = clz.getDeclaredConstructors();
for (Constructor c : declaredConstructors) {
System.out.println(c);
System.out.println(Arrays.toString(c.getParameterTypes()));
System.out.println(c.getDeclaringClass());
}
/**
public 面试题.User(java.lang.String,int,int)
[class java.lang.String, int, int]
class 面试题.User
public 面试题.User(java.lang.String)
[class java.lang.String]
class 面试题.User
private 面试题.User()
[]
class 面试题.User
*/
}

/**
* 获取指定参数类型的构造器,和访问权限无关
* @throws NoSuchMethodException
*/
private static void getDeclaredConstructor() throws NoSuchMethodException {
Constructor<User> declaredConstructor = clz.getDeclaredConstructor(String.class);
System.out.println(declaredConstructor); // private 面试题.User(java.lang.String)
}

private static void invokeConstructor() throws Exception {
// 这种newInstance方法只能作用于 public 的 无参构造方法,局限性太大
User user1 = clz.newInstance();
System.out.println(user1);

Constructor<User> constructor1 = clz.getDeclaredConstructor();
User user2 = constructor1.newInstance();
System.out.println(user2);

Constructor<User> constructor2 = clz.getDeclaredConstructor(String.class);
constructor2.setAccessible(true); // 不加这句就实例化不了私有(private)的构造器
User user3 = constructor2.newInstance("呵呵");
System.out.println(user3);

Constructor<User> constructor3 = clz.getDeclaredConstructor(String.class, int.class, int.class);
User user4 = constructor3.newInstance("哈哈", 22, 2);
System.out.println(user4);

/**
public User()
User -> name: 无名氏 age: 无年龄 sex: 无性别
public User()
User -> name: 无名氏 age: 无年龄 sex: 无性别
private User(String name): 呵呵
User -> name: 呵呵 age: 无年龄 sex: 无性别
public User(String name, int age, int sex): 哈哈 22 2
User -> name: 哈哈 age: 22 sex: 女
*/
}

/**
* getDeclaredMethods和getMethods区别
* getMethods获得包括自身和继承过来的所有public方法
* getDeclaredMathods获取自身所有的方法,不包括继承的,和访问权限无关
*/
private static void getAllMethods() {
Method[] methods = clz.getDeclaredMethods();
for (Method m : methods) {
System.out.println(m);
}
/**
public java.lang.String 面试题.User.toString()
public java.lang.String 面试题.User.getName()
public void 面试题.User.setName(java.lang.String)
public int 面试题.User.getAge()
public void 面试题.User.setAge(int)
public int 面试题.User.getSex()
public void 面试题.User.setSex(int)
private void 面试题.User.sayHello()
public void 面试题.User.sayHi(java.lang.String)
*/
}

/**
* 只有通过方法签名才能找到唯一的方法
* 方法名 + 参数列表
* 得到私有方法必须使用 getDeclaredMethod
*/
private static void getOneMethod() throws NoSuchMethodException {
Method method1 = clz.getMethod("sayHi", String.class);
System.out.println(method1); // public void 面试题.User.sayHi(java.lang.String)

Method method2 = clz.getDeclaredMethod("sayHello");
System.out.println(method2); // private void 面试题.User.sayHello()
}

private static void invokeMethod() throws NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
Method method = clz.getMethod("getMoney", String.class);
double money = (Double)method.invoke(clz.newInstance(), "benjamin");
System.out.println(money);
/**
public User()
get money 200.12 for benjamin
200.12
*/
}

/**
* 调用静态方法
* 如果方法是静态的,那么可以忽略指定的obj参数,将obj参数设为null即可
*/
private static void invokeStaticMethod() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Method method = clz.getMethod("staticMethod");
method.invoke(null); // invoke static method...
}

/**
* 调用可变参数的静态方法
*/
private static void invokeVariableLengthMethod() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Method method = clz.getMethod("show", int[].class);
int result = (Integer)method.invoke(null, new int[]{2, 4});
System.out.println(result); // 6

Method method2 = clz.getMethod("show2", String[].class);
// 下面这么写会报Exception in thread "main" java.lang.IllegalArgumentException: wrong number of arguments
// 基本类型和引用类型不同,引用类型会自动解包
// 对于数组类型的引用类型的参数,底层会自动进行解包,为了解决该问题,我们使用Object的一维数组把实际参数包装起来
// method2.invoke(null, new String[]{"A", "B", "C"});

method2.invoke(null, new Object[]{new String[]{"A", "B", "C"}}); // [A, B, C]
}

/**
* 获取所有字段,包括私有的
* 如果想只获取public的字段以及父类的public字段,使用getFields()方法
*/
private static void getAllFields() {
Field[] declaredFields = clz.getDeclaredFields();
for (Field field : declaredFields) {
System.out.println(field);
}
/**
private java.lang.String 面试题.User.name
private int 面试题.User.age
private int 面试题.User.sex
*/
}

/**
* 设置字段的值
*/
private static void invokeField() throws NoSuchFieldException, IllegalAccessException, InstantiationException {
Field ageField = clz.getDeclaredField("age");
ageField.setAccessible(true);
Object obj = clz.newInstance();
ageField.set(obj, 12);
System.out.println(obj);
/**
public User()
User -> name: 无名氏 age: 12 sex: 无性别
*/
}

/**
* 获取字段值
*/
private static void getFieldValue() throws NoSuchFieldException, IllegalAccessException, InstantiationException {
Field ageField = clz.getDeclaredField("age");
ageField.setAccessible(true);
Object obj = clz.newInstance();
Object age = ageField.get(obj);
System.out.println(age);
/**
public User()
0
*/
}

private static void getOtherAPI() {
int modifiers = clz.getModifiers();
System.out.println(modifiers); // 1
System.out.println(Modifier.toString(modifiers)); // public

System.out.println(clz.getName()); // 面试题.User
System.out.println(clz.getSimpleName()); // User
System.out.println(clz.getPackage()); // package 面试题
System.out.println(clz.getSuperclass()); // class java.lang.Object
System.out.println(clz.isArray()); // false
System.out.println(clz.isEnum()); // false
}

/**
* 调用泛型参数的方法
*/
private static void invokeGenericMethod() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Class clz = Arrays.class;
Method asListMethod = clz.getDeclaredMethod("asList", Object[].class);
List result = (List) asListMethod.invoke(null, new Object[]{new Object[]{"A", "B", "C"}});
System.out.println(result); // [A, B, C]
}

/**
* 得到泛型字段的类型
*/
private static void getGenericFieldType() throws NoSuchFieldException {
Field cacheField = ooxxClass.getDeclaredField("cache");
Type genericType = cacheField.getGenericType();
System.out.println(genericType); // java.util.Map<java.lang.String, java.lang.Object>
ParameterizedType parameterizedType = (ParameterizedType) genericType;
Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
System.out.println(Arrays.toString(actualTypeArguments)); // [class java.lang.String, class java.lang.Object]
}
}
目录
相关文章
|
2月前
|
存储 Java
[Java]反射
本文详细介绍了Java反射机制的基本概念、使用方法及其注意事项。首先解释了反射的定义和类加载过程,接着通过具体示例展示了如何使用反射获取和操作类的构造方法、方法和变量。文章还讨论了反射在类加载、内部类、父类成员访问等方面的特殊行为,并提供了通过反射跳过泛型检查的示例。最后,简要介绍了字面量和符号引用的概念。全文旨在帮助读者深入理解反射机制及其应用场景。
22 0
[Java]反射
|
3月前
|
安全 Java 索引
Java——反射&枚举
本文介绍了Java反射机制及其应用,包括获取Class对象、构造方法、成员变量和成员方法。反射允许在运行时动态操作类和对象,例如创建对象、调用方法和访问字段。文章详细解释了不同方法的使用方式及其注意事项,并展示了如何通过反射获取类的各种信息。此外,还介绍了枚举类型的特点和使用方法,包括枚举的构造方法及其在反射中的特殊处理。
66 9
Java——反射&枚举
|
2月前
|
安全 Java 测试技术
🌟Java零基础-反射:从入门到精通
【10月更文挑战第4天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
26 2
|
3月前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
2月前
|
IDE Java 编译器
java的反射与注解
java的反射与注解
17 0
|
3月前
|
Java 程序员 编译器
Java的反射技术reflect
Java的反射技术允许程序在运行时动态加载和操作类,基于字节码文件构建中间语言代码,进而生成机器码在JVM上执行,实现了“一次编译,到处运行”。此技术虽需更多运行时间,但广泛应用于Spring框架的持续集成、动态配置及三大特性(IOC、DI、AOP)中,支持企业级应用的迭代升级和灵活配置管理,适用于集群部署与数据同步场景。
|
3月前
|
存储 安全 Java
扫盲java基础-反射(一)
扫盲java基础-反射(一)
|
3月前
|
Java
扫盲java基础-反射(二)
扫盲java基础-反射(二)
|
5月前
|
安全 Java 测试技术
day26:Java零基础 - 反射
【7月更文挑战第26天】🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
35 5
|
4月前
|
缓存 安全 Java
【Java 第十篇章】反射
Java 反射技术让程序能在运行时动态获取类信息并操作对象,极大提升了灵活性与扩展性。本文将介绍反射的基本概念、原理及应用,包括如何使用 `Class`、`Field`、`Method` 和 `Constructor` 类进行动态操作。此外,还将探讨反射在动态加载、框架开发与代码测试中的应用场景,并提醒开发者注意性能与安全方面的问题,帮助你更合理地运用这一强大工具。
31 0
下一篇
无影云桌面