java反射机制小结

java的反射机制一直是java的一块重要内容,掌握它对日后的框架学习会有很大的帮助。 

我们先来看看什么是反射机制。反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(self-representation)和监测(examination),并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。Reflection 是 Java 程序开发语言的特征之一,它允许运行中的 Java 程序对自身进行检查,或者说“自审”,并能直接操作程序的内部属性。

那么Java反射有什么作用呢?

假如我们有两个程序员,一个程序员在写程序的时候,需要使用第二个程序员所写的类,但第二个程序员并没完成他所写的类。那么第一个程序员的代码能否通过编译呢?这

是不能通过编译的。利用Java反射的机制,就可以让第一个程序员在没有得到第二个程序员所写的类的时候,来完成自身代码的编译。

Java的反射机制它知道类的基本结构,这种对Java类结构探知的能力,我们称为Java类的“自审”。大家都用

过Jcreator和eclipse。当我们构建出一个对象的时候,去调用该对象的方法和属性的时候。一按点,编译工具

就会自动的把该对象能够使用的所有的方法和属性全部都列出来,供用户进行选择。这就是利用了Java反射的

原理,是对我们创对象的探知、自审。

java的反射机制就是增加程序的灵活性,避免将程序写死到代码里,

例如: 实例化一个 person()对象, 不使用反射,new person(); 如果想变成 实例化 其他类, 那么必须修改源代码,并重新编译。

使用反射:class.forName("person").newInstance(); 而且这个类描述可以写到配置文件中,如 **.xml, 这样如果想实例化其他类,只要修改配置文件的"类描述"就可以了,不

需要重新修改代码并编译。

由上述内容可知,很多时候反射并非一定要用,但框架对灵活性要求较高,所以反射在框架中使用较多。


最后我给出反射的一些简单使用方法。

先在cla包下定义一个类Foo,我要获得该类的类类型(class type)。哦,这里要补充一下,在面向对象的世界里,万物

皆对象。类也是一种对象,它是java.lang.Class类的对象。Class类只能由虚拟机创建。

有以下三种方法获得该类的类类型:

第一种:Class class1 = Foo.class;

第二种:Class class2 = foo1.getClass();  //其中foo1是Foo的一个实例对象

第三种:

Class class3 = null;
try {
class3 = Class.forName("cla.Foo");
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

有了类类型,我们就有了该类编译后的信息。同一个类编译后只产生一个实例。所以class1 == class2 == class3

当然,class类还有很多方法。如getName()(包名+类名)。getMethods()获得所有能在类外部访问到的方法。getDeclaredMethods()获得所有声明的方法。类似的还有

getFields(),getDeclaredFields()获得成员变量信息。getConstructors(),getDeclaredConstructors()获得构造函数信息等等。

直接贴个示例代码:

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;


public class ClassUtil {

	public static void printClassMethod(Object object) {
		Class class1 = object.getClass();
		System.out.println("类的名称是:" + class1.getName());
		
		//包括父类继承来的方法
		Method[] methods = class1.getMethods();
		//不包括父类继承的方法
//		Method[] methods = class1.getDeclaredMethods();
		
		for(Method method : methods) {
			System.out.print(method.getReturnType() + " ");
			System.out.print(method.getName() + "(");
			
			Class[] paramTypes = method.getParameterTypes();
			for(Class param : paramTypes) {
				System.out.print(param.getName() + " ");
			}
			System.out.println(")");
		}
		
	}
	private static void getClassField(Object object) {
		Class class1 = object.getClass();
		Field[] fs = class1.getFields();
//		Field[] fs = class1.getDeclaredFields();
		for(Field fd : fs) {
			Class fieldType = fd.getType();
			String typeName = fieldType.getName();
			//获得成员变量的名称
			String fieldName = fd.getName();
			System.out.println(typeName + " " + fieldName);
		}
	}
	
	public static void printConMessage(Object object) {
		Class class1 = object.getClass();
//		Constructor[] constructors = class1.getConstructors();
		Constructor[] constructors = class1.getDeclaredConstructors();
		
		for(Constructor constructor : constructors) {
			System.out.print(constructor.getName() + "(");
			
			Class[] paramTypes = constructor.getParameterTypes();
			for(Class param : paramTypes) {
				System.out.print(param.getName() + " ");
			}
			System.out.println(")");
		}
		
	}
	
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Father father = new Father();
		Son son = new Son(1, "gg", 9.9);
		
		printClassMethod(father);
		System.out.println();
		printClassMethod(son);
		
		/*getClassField(father);
		System.out.println();
		getClassField(son);*/
		
		/*printConMessage(father);
		System.out.println();
		printConMessage(son);*/
	}

}
说到这我举个反射应用的小栗子:

最后贴一个调用方法的栗子:

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;


public class MethodDemo1 {

	public static void main(String[] args) 
throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
		// TODO Auto-generated method stub
		A a = new A();
		Class class1 = a.getClass();
		/**
		 * getMethod方法最后一个参数是可变参数 这里亦可写为("print",  int.class, int.class);
		 */
		Method method = class1.getMethod("print",  new Class[]{int.class, int.class});
		method.invoke(a, 10,20);
		
		method = class1.getDeclaredMethod("print",  String.class,String.class);
		method.invoke(a, "HHgg", "hhHH");
	}

}

class A {
	public void print(int a,int b) {
		System.out.println(a+b);
	}
	
	public void print(String a, String b) {
		System.out.println(a.toUpperCase() + "," + b.toLowerCase());
	}
}

通过反射我们还可以验证java的伪泛型,只要绕过编译,集合的泛型都是去泛型化的。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值