Java反射机制详解.docx
- 文档编号:6657786
- 上传时间:2023-01-08
- 格式:DOCX
- 页数:15
- 大小:53.33KB
Java反射机制详解.docx
《Java反射机制详解.docx》由会员分享,可在线阅读,更多相关《Java反射机制详解.docx(15页珍藏版)》请在冰豆网上搜索。
Java反射机制详解
首先,我们在开始前提出一个问题:
1.在运行时,对于一个java类,能否知道属性和方法;能否去调用它的任意方法?
答案是肯定的。
本节所有目录如下:
1.什么是JAVA的反射机制
2.JDK中提供的ReflectionAPI
3.JAVA反射机制提供了什么功能
∙获取类的Class对象
∙获取类的Fields
∙获取类的Method
∙获取类的Constructor
∙新建类的实例
Class
通过Constructor对象的方法newInstance
4.调用类的函数
调用private函数
5.设置/获取类的属性值
private属性
6.动态创建代理类
动态代理源码分析
7.JAVA反射Class
8.JAVA反射原理分析
Class文件结构
JVM加载类对象,对反射的支持
9.JAVA反射的应用
一、什么是JAVA的反射机制
Java反射是Java被视为动态(或准动态)语言的一个关键性质。
这个机制允许程序在运行时透过ReflectionAPIs取得任何一个已知名称的class的内部信息,包括其modifiers(诸如public,static等)、superclass(例如Object)、实现之interfaces(例如Cloneable),也包括fields和methods的所有信息,并可于运行时改变fields内容或唤起methods。
Java反射机制容许程序在运行时加载、探知、使用编译期间完全未知的classes。
换言之,Java可以加载一个运行时才得知名称的class,获得其完整结构。
二、JDK中提供的ReflectionAPI
Java反射相关的API在包java.lang.reflect中,JDK1.6.0的reflect包如下图:
Member接口
该接口可以获取有关类成员(域或者方法)后者构造函数的信息。
AccessibleObject类
该类是域(field)对象、方法(method)对象、构造函数(constructor)对象的基础类。
它提供了将反射的对象标记为在使用时取消默认Java语言访问控制检查的能力。
Array类
该类提供动态地生成和访问JAVA数组的方法。
Constructor类
提供一个类的构造函数的信息以及访问类的构造函数的接口。
Field类
提供一个类的域的信息以及访问类的域的接口。
Method类
提供一个类的方法的信息以及访问类的方法的接口。
Modifier类
提供了static方法和常量,对类和成员访问修饰符进行解码。
Proxy类
提供动态地生成代理类和类实例的静态方法。
三、JAVA反射机制提供了什么功能
Java反射机制提供如下功能:
在运行时判断任意一个对象所属的类
在运行时构造任意一个类的对象
在运行时判段任意一个类所具有的成员变量和方法
在运行时调用任一个对象的方法
在运行时创建新类对象
在使用Java的反射功能时,基本首先都要获取类的Class对象,再通过Class对象获取其他的对象。
这里首先定义用于测试的类:
[java] viewplaincopy
1.class Type{
2. public int pubIntField;
3. public String pubStringField;
4. private int prvIntField;
5.
6. public Type(){
7. Log("Default Constructor");
8. }
9.
10. Type(int arg1, String arg2){
11. pubIntField = arg1;
12. pubStringField = arg2;
13.
14. Log("Constructor with parameters");
15. }
16.
17. public void setIntField(int val) {
18. this.prvIntField = val;
19. }
20. public int getIntField() {
21. return prvIntField;
22. }
23.
24. private void Log(String msg){
25. System.out.println("Type:
" + msg);
26. }
27.}
28.
29.class ExtendType extends Type{
30. public int pubIntExtendField;
31. public String pubStringExtendField;
32. private int prvIntExtendField;
33.
34. public ExtendType(){
35. Log("Default Constructor");
36. }
37.
38. ExtendType(int arg1, String arg2){
39. pubIntExtendField = arg1;
40. pubStringExtendField = arg2;
41.
42. Log("Constructor with parameters");
43. }
44.
45. public void setIntExtendField(int field7) {
46. this.prvIntExtendField = field7;
47. }
48. public int getIntExtendField() {
49. return prvIntExtendField;
50. }
51.
52. private void Log(String msg){
53. System.out.println("ExtendType:
" + msg);
54. }
55.}
1、获取类的Class对象
Class类的实例表示正在运行的Java应用程序中的类和接口。
获取类的Class对象有多种方式:
调用getClass
Booleanvar1=true;
Class
>classType2=var1.getClass();
System.out.println(classType2);
输出:
classjava.lang.Boolean
运用.class语法
Class
>classType4=Boolean.class;
System.out.println(classType4);
输出:
classjava.lang.Boolean
运用staticmethodClass.forName()
Class
>classType5=Class.forName("java.lang.Boolean");
System.out.println(classType5);
输出:
classjava.lang.Boolean
运用primitivewrapperclasses的TYPE语法
这里返回的是原生类型,和Boolean.class返回的不同
Class
>classType3=Boolean.TYPE;
System.out.println(classType3);
输出:
boolean
2、获取类的Fields
可以通过反射机制得到某个类的某个属性,然后改变对应于这个类的某个实例的该属性值。
JAVA的Class
public FieldgetField(String name)
返回一个Field对象,它反映此Class对象所表示的类或接口的指定公共成员字段
public Field[]getFields()
返回一个包含某些Field对象的数组,这些对象反映此Class对象所表示的类或接口的所有可访问公共字段
public FieldgetDeclaredField(Stringname)
返回一个Field对象,该对象反映此Class对象所表示的类或接口的指定已声明字段
public Field[]getDeclaredFields()
返回Field对象的一个数组,这些对象反映此Class对象所表示的类或接口所声明的所有字段
[java] viewplaincopy
1.Class
> classType = ExtendType.class;
2.
3.// 使用getFields获取属性
4.Field[] fields = classType.getFields();
5.for (Field f :
fields)
6.{
7. System.out.println(f);
8.}
9.
10.System.out.println();
11.
12.// 使用getDeclaredFields获取属性
13.fields = classType.getDeclaredFields();
14.for (Field f :
fields)
15.{
16. System.out.println(f);
17.}
输出:
publicintcom.quincy.ExtendType.pubIntExtendField
publicjava.lang.Stringcom.quincy.ExtendType.pubStringExtendField
publicintcom.quincy.Type.pubIntField
publicjava.lang.Stringcom.quincy.Type.pubStringField
publicintcom.quincy.ExtendType.pubIntExtendField
publicjava.lang.Stringcom.quincy.ExtendType.pubStringExtendField
privateintcom.quincy.ExtendType.prvIntExtendField
可见getFields和getDeclaredFields区别:
getFields返回的是申明为public的属性,包括父类中定义,
getDeclaredFields返回的是指定类定义的所有定义的属性,不包括父类的。
3、获取类的Method
通过反射机制得到某个类的某个方法,然后调用对应于这个类的某个实例的该方法
Class
public MethodgetMethod(String name,Class
>...parameterTypes)
返回一个Method对象,它反映此Class对象所表示的类或接口的指定公共成员方法
public Method[]getMethods()
返回一个包含某些Method对象的数组,这些对象反映此Class对象所表示的类或接口(包括那些由该类或接口声明的以及从超类和超接口继承的那些的类或接口)的公共member方法
public MethodgetDeclaredMethod(Stringname,Class
>...parameterTypes)
返回一个Method对象,该对象反映此Class对象所表示的类或接口的指定已声明方法
public Method[]getDeclaredMethods()
返回Method对象的一个数组,这些对象反映此Class对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法
[java] viewplaincopy
1.// 使用getMethods获取函数
2.Class
> classType = ExtendType.class;
3.Method[] methods = classType.getMethods();
4.for (Method m :
methods)
5.{
6. System.out.println(m);
7.}
8.
9.System.out.println();
10.
11.// 使用getDeclaredMethods获取函数
12.methods = classType.getDeclaredMethods();
13.for (Method m :
methods)
14.{
15. System.out.println(m);
16.}
输出:
publicvoidcom.quincy.ExtendType.setIntExtendField(int)
publicintcom.quincy.ExtendType.getIntExtendField()
publicvoidcom.quincy.Type.setIntField(int)
publicintcom.quincy.Type.getIntField()
publicfinalnativevoidjava.lang.Object.wait(long)throwsjava.lang.InterruptedException
publicfinalvoidjava.lang.Object.wait()throwsjava.lang.InterruptedException
publicfinalvoidjava.lang.Object.wait(long,int)throwsjava.lang.InterruptedException
publicbooleanjava.lang.Object.equals(java.lang.Object)
publicjava.lang.Stringjava.lang.Object.toString()
publicnativeintjava.lang.Object.hashCode()
publicfinalnativejava.lang.Classjava.lang.Object.getClass()
publicfinalnativevoidjava.lang.Object.notify()
publicfinalnativevoidjava.lang.Object.notifyAll()
privatevoidcom.quincy.ExtendType.Log(java.lang.String)
publicvoidcom.quincy.ExtendType.setIntExtendField(int)
publicintcom.quincy.ExtendType.getIntExtendField()
4、获取类的Constructor
通过反射机制得到某个类的构造器,然后调用该构造器创建该类的一个实例
Class
public Constructor
>...parameterTypes)
返回一个Constructor对象,它反映此Class对象所表示的类的指定公共构造方法
public Constructor
>[]getConstructors()
返回一个包含某些Constructor对象的数组,这些对象反映此Class对象所表示的类的所有公共构造方法
public Constructor
>...parameterTypes)
返回一个Constructor对象,该对象反映此Class对象所表示的类或接口的指定构造方法
public Constructor
>[]getDeclaredConstructors()
返回Constructor对象的一个数组,这些对象反映此Class对象表示的类声明的所有构造方法。
它们是公共、保护、默认(包)访问和私有构造方法
[java] viewplaincopy
1.// 使用getConstructors获取构造器
2.Constructor
>[] constructors = classType.getConstructors();
3.for (Constructor
> m :
constructors)
4.{
5. System.out.println(m);
6.}
7.
8.System.out.println();
9.
10.// 使用getDeclaredConstructors获取构造器
11.constructors = classType.getDeclaredConstructors();
12.for (Constructor
> m :
constructors)
13.{
14. System.out.println(m);
15.}
16.
17.输出:
18.public com.quincy.ExtendType()
19.
20.public com.quincy.ExtendType()
21.com.quincy.ExtendType(int,java.lang.String)
5、新建类的实例
通过反射机制创建新类的实例,有几种方法可以创建
调用无自变量ctor
1、调用类的Class对象的newInstance方法,该方法会调用对象的默认构造器,如果没有默认构造器,会调用失败.
Class
>classType=ExtendType.class;
Objectinst=classType.newInstance();
System.out.println(inst);
输出:
Type:
DefaultConstructor
ExtendType:
DefaultConstructor
com.quincy.ExtendType@d80be3
2、调用默认Constructor对象的newInstance方法
Class
>classType=ExtendType.class;
Constructor
>constructor1=classType.getConstructor();
Objectinst=constructor1.newInstance();
System.out.println(inst);
输出:
Type:
DefaultConstructor
ExtendType:
DefaultConstructor
com.quincy.ExtendType@1006d75
调用带参数ctor
3、调用带参数Constructor对象的newInstance方法
Constructor
>constructor2=
classType.getDeclaredConstructor(int.class,String.class);
Objectinst=constructor2.newInstance(1,"123");
System.out.println(inst);
输出:
Type:
DefaultConstructor
ExtendType:
Constructorwithparameters
com.quincy.ExtendType@15e83f9
6、调用类的函数
通过反射获取类Method对象,调用Field的Invoke方法调用函数。
[java] viewplaincopy
1.Class
> classType = ExtendType.class;
2.Object inst = classType.newInstance();
3.Method logMethod = classType.getDeclaredMethod("Log", String.class);
4.logMethod.invoke(inst, "test");
5.
6.输出:
7.Type:
Default Constructor
8.ExtendType:
Default Constructor
9.Class com.quincy.ClassT can not access a member of class com.quincy.ExtendType with modifiers "private"
10.
11.上面失败是由于没有权限调用private函数,这里需要设置Accessible为true;
12.Class
> classType = ExtendType.class;
13.Object inst = classType.newInstance();
14.Method logMethod = classType.getDeclaredMethod("Log", String.cla
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Java 反射 机制 详解
![提示](https://static.bdocx.com/images/bang_tan.gif)