本篇将从以下几个方面讲述反射的知识:
- class 的使用
- 方法的反射
- 构造函数的反射
- 成员变量的反射
一、什么是class类
在面向对象的世界里,万物皆对象。类是对象,类是java.lang.Class类的实例对象。另外class类只有java虚拟机才能new出来。任何一个类都是Class 类的实例对象。这实例对象有三种表达方式:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class User{ } public class ClassTest{ User u= new User(); //方式1: Class c1=User. class ; //方式2: Class c2=u.getClass(); //方式3: Class c3=Class.forName( "com.forezp.User" ); //可以通过类的类型创建该类的实例对象 User user=(User)c1.newInstance(); } |
二、class类的动态加载
Class.forName(类的全称);该方法不仅表示了类的类型,还代表了动态加载类。编译时刻加载类是静态加载、运行时刻加载类是动态加载类。
三、获取方法信息
基本的数据类型,void关键字都Class 类的实例;可以通过getame();getSimpleName()获取类的名称。
1
2
3
4
5
|
Class c1=String. class ; Class c2= int . class ; Class c3= void . class ; System.out.println(c1.getName()); System.out.println(c2.getSimpleName()); |
获取类的所有方法,并打印出来:
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
|
public static void printClassInfo(Object object){ Class c=object.getClass(); System.out.println( "类的名称:" +c.getName()); /** * 一个成员方法就是一个method对象 * getMethod()所有的 public方法,包括父类继承的 public * getDeclaredMethods()获取该类所有的方法,包括private ,但不包括继承的方法。 */ Method[] methods=c.getMethods(); //获取方法 //获取所以的方法,包括private ,c.getDeclaredMethods(); for ( int i= 0 ;i<methods.length;i++){ //得到方法的返回类型 Class returnType=methods[i].getReturnType(); System.out.print(returnType.getName()); //得到方法名: System.out.print(methods[i].getName()+ "(" ); Class[] parameterTypes=methods[i].getParameterTypes(); for (Class class1:parameterTypes){ System.out.print(class1.getName()+ "," ); } System.out.println( ")" ); } } |
运行:
1
2
3
4
5
|
类的名称:java.lang.String booleanequals(java.lang.Object,) java.lang.StringtoString() inthashCode() … |
四、获取成员变量的信息
也可以获取类的成员变量信息
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
public static void printFiledInfo(Object o){ Class c=o.getClass(); /** * getFileds()获取public * getDeclaredFields()获取所有 */ Field[] fileds=c.getDeclaredFields(); for (Field f:fileds){ //获取成员变量的类型 Class filedType=f.getType(); System.out.println(filedType.getName()+ " " +f.getName()); } } |
1
2
3
4
5
|
public static void main(String[] args){ String s= "ss" ; //ClassUtil.printClassInfo(s); ClassUtil.printFiledInfo(s); } |
运行:
1
2
3
4
5
6
7
|
[C value int hash long serialVersionUID [Ljava.io.ObjectStreamField; serialPersistentFields java.util.Comparator CASE_INSENSITIVE_ORDER int HASHING_SEED int hash32 |
五、获取构造函数的信息
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public static void printConstructInfo(Object o){ Class c=o.getClass(); Constructor[] constructors=c.getDeclaredConstructors(); for (Constructor con:constructors){ System.out.print(con.getName()+”(“); Class[] typeParas=con.getParameterTypes(); for (Class class1:typeParas){ System.out.print(class1.getName()+” ,”); } System.out.println(“)”); } } |
1
2
3
4
5
6
|
public static void main(String[] args){ String s= "ss" ; //ClassUtil.printClassInfo(s); //ClassUtil.printFiledInfo(s); ClassUtil.printConstructInfo(s); } |
运行:
1
2
3
4
5
6
7
8
9
10
|
java.lang.String([B ,) java.lang.String([B , int , int ,) java.lang.String([B ,java.nio.charset.Charset ,) java.lang.String([B ,java.lang.String ,) java.lang.String([B , int , int ,java.nio.charset.Charset ,) java.lang.String( int , int ,[C ,) java.lang.String([C , boolean ,) java.lang.String(java.lang.StringBuilder ,) java.lang.String(java.lang.StringBuffer ,) ... |
六、方法反射的操作
获取一个方法:需要获取方法的名称和方法的参数才能决定一个方法。
方法的反射操作:
1
|
method.invoke(对象,参数列表); |
举个例子:
1
2
3
4
5
6
7
8
9
10
|
class A{ public void add( int a, int b){ System.out.print(a+b); } public void toUpper(String a){ System.out.print(a.toUpperCase()); } } |
1
2
3
4
5
6
7
8
9
10
11
12
|
public static void main(String[] args) { A a= new A(); Class c=a.getClass(); try { Method method=c.getMethod( "add" , new Class[]{ int . class , int . class }); //也可以 Method method=c.getMethod("add",int.class,int.class); //方法的反射操作 method.invoke(a, 10 , 10 ); } catch (Exception e){ e.printStackTrace(); } } |
运行:
1
|
|
本篇文章已经讲解了java反射的基本用法, 它可以在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!