菜鸟学Java——Java反射机制(一)

说到反射,相信有过编程经验的人都不会陌生。反射机制让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

Type

Type 是 Java 编程语言中所有类型的公共高级接口。

TypeVariable

TypeVariable 是各种类型变量的公共高级接口。

WildcardType

WildcardType 表示一个通配符类型表达式,如 ?、? extends Number 或 ? super Integer。

 

类摘要

AccessibleObject

AccessibleObject 类是 Field、Method 和 Constructor 对象的基类。

Array

Array 类提供了动态创建和访问 Java 数组的方法。

Constructor

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反射机制(二)》

 

 

 

你可能感兴趣的:(Java)