说到反射,相信有过编程经验的人都不会陌生。反射机制让Java变得更加的灵活。反射机制在Java的众多特性中是非常重要的一个。下面就让我们一点一点了解它是怎么一回事。
什么是反射
在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
反射有什么用
1,在运行时判断任意一个对象所属的类;
2,在运行时构造任意一个类的对象;
3,在运行时判断任意一个类所具有的成员变量和方法;
4,在运行时调用任意一个对象的方法;
5,生成动态代理。
反射的API
学习一门语言,一种比较好的方式就是看它的API文档,因为它的API是最权威的,最准确的。下面我们一起看看JDK的API对于反射是如何描述的吧!
与反射有关的所有接口以及类都在java.lang.reflect包里。
接口
接口摘要 |
|
AnnotatedElement |
表示目前正在此 VM 中运行的程序的一个已注释元素。 |
GenericArrayType |
GenericArrayType 表示一种数组类型,其组件类型为参数化类型或类型变量。 |
GenericDeclaration |
声明类型变量的所有实体的公共接口。 |
InvocationHandler |
InvocationHandler 是代理实例的调用处理程序 实现的接口。 |
Member |
成员是一种接口,反映有关单个成员(字段或方法)或构造方法的标识信息。 |
ParameterizedType |
ParameterizedType 表示参数化类型,如 Collection<String>。 |
Type |
Type 是 Java 编程语言中所有类型的公共高级接口。 |
TypeVariable<D extends GenericDeclaration> |
TypeVariable 是各种类型变量的公共高级接口。 |
WildcardType |
WildcardType 表示一个通配符类型表达式,如 ?、? extends Number 或 ? super Integer。 |
类
类摘要 |
|
AccessibleObject |
AccessibleObject 类是 Field、Method 和 Constructor 对象的基类。 |
Array |
Array 类提供了动态创建和访问 Java 数组的方法。 |
Constructor<T> |
Constructor 提供关于类的单个构造方法的信息以及对它的访问权限。 |
Field |
Field 提供有关类或接口的单个字段的信息,以及对它的动态访问权限。 |
Method |
Method 提供关于类或接口上单独某个方法(以及如何访问该方法)的信息。 |
Modifier |
Modifier 类提供了 static 方法和常量,对类和成员访问修饰符进行解码。 |
Proxy |
Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。 |
ReflectPermission |
反射操作的 Permission 类。 |
下面给大家写了一个小实例:
package reflection; public class UserInfo { private Integer id; private String userName; private String password; public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } @Override public String toString(){ return this.getClass().getName(); } }
package reflection; public class GetClass { public boolean ClassCheck(){ try { System.out.println("第一种,通过类本身获得对象"); Class UserClass = this.getClass(); System.out.println("第一种方式成功!类名:"+UserClass.toString()+"\n"); System.out.println("第二种,通过子类实例获取父类"); UserInfo ui = new UserInfo(); UserClass = ui.getClass(); Class SubUserClass = UserClass.getSuperclass(); System.out.println("第二种方式成功!类名:"+SubUserClass.toString()+"\n"); System.out.println("第三种,通过类名加.class获得对象"); Class ForClass = reflection.UserInfo.class; System.out.println("第三种方式成功!类名:"+ForClass.toString()+"\n"); System.out.println("第四种,通过类名的字符串获得对象"); Class ForName = Class.forName("reflection.UserInfo"); System.out.println("第四种方式成功!类名:"+ForName.toString()+"\n"); } catch (Exception e) { e.printStackTrace(); return false; } return true; } @Override public String toString(){ return this.getClass().getName(); } public static void main(String[] args) { GetClass gc = new GetClass(); gc.ClassCheck(); if (gc.ClassCheck()) { System.out.println("所有反射全部成功!"); } else { System.out.println("反射有问题,请检查!"); } } }
上面四种方法里面,我用的最多的是第四种,这种一般都是配合配置文件一起用的。反射与配置文件的结合让我们的代码变得非常的灵活,易扩展、以维护。可谓是好处多多呀,所以才有那句”反射反射,程序员的快乐!“一起尽情的快乐吧!
PS:下一篇将结合实例更加详细的讲解!《菜鸟学编程(十)——Java反射机制(二)》