黑马程序员——反射

------<a href="http://www.itheima.com" target="blank">Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流! -------

反射就是把Java类中的各种成分映射成相应的java类。

一、Class类(反射的基石

import java.util.Date;

public class ReflectPoint {
	private Date birthday = new Date();
	
	private int x;
	public int y;
	public String str1 = "ball";
	public String str2 = "basketball";
	public String str3 = "itcast";
	
	public ReflectPoint(int x, int y) {
		super();
		this.x = x;
		this.y = y;
	}
	
	
	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + x;
		result = prime * result + y;
		return result;
	}


	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		final ReflectPoint other = (ReflectPoint) obj;
		if (x != other.x)
			return false;
		if (y != other.y)
			return false;
		return true;
	}


	@Override
	public String toString(){
		return str1 + ":" + str2 + ":" + str3;
	}


	public int getX() {
		return x;
	}


	public void setX(int x) {
		this.x = x;
	}


	public int getY() {
		return y;
	}


	public void setY(int y) {
		this.y = y;
	}


	public Date getBirthday() {
		return birthday;
	}


	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}
	
}


1、Class类(反射的基石)

获得指定名字的Class对象: Class<?> forName(String className)
获得注解对:Annotation[] getAnnotations()
获得类加载器:ClassLoader getClassLoader()
获取构造方法:Constructor<T> getConstructor(Class<?>... parameterTypes),Constructor<?>[] getConstructors()(获得所有构造方法),Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)(无视权限 暴力反射前提),Constructor<?>[] getDeclaredConstructors()(无视权限 暴力反射前提)。
获得成员变量:Field getField(String name),Field[] getFields()(获得所有成员变量),Field getDeclaredField(String name)(无视权限 暴力反射前提), Field[] getDeclaredFields()  (无视权限 暴力反射前提)。
获得方法: Method getMethod(String name,Class<?>... parameterTypes),Method[] getMethods()(获得所有方法), Method getDeclaredMethod(String name, Class<?>... parameterTypes)  (无视权限 暴力反射前提), Method[] getDeclaredMethods()  (无视权限 暴力反射前提)
获得父类:Class<? super T> getSuperclass()
创建实例对象: T newInstance()
加载配置文件:InputStream getResourceAsStream(String name)

2、获取.class字节码

(1)类名.class,例如,System.class,(2)对象.getClass(),例如,new Date().getClass(),(3)Class.forName("类名"),例如,Class.forName("java.util.Date");,(4)九个预定义Class实例对象,(5)数组类型的Class实例对象,Class.isArray()。

3、Field类(代表某个类中的一个成员变量)

(1)通过Class类获取实例对象;(2)获取成员变量的值Object get(Object obj);(3)设置成员变量的值 set(Object obj,setInt(基本数据类型)(Object obj, int i) 
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Arrays;

public class ReflectTest {

	/**
	 * @param args
	 */
	public static void main(String[] args) throws Exception {
		// TODO Auto-generated method stub
		String str1 = "abc";
		Class cls1 = str1.getClass();
		Class cls2 = String.class;
		Class cls3 = Class.forName("java.lang.String");
		System.out.println(cls1 == cls2);
		System.out.println(cls1 == cls3);
		
		System.out.println(cls1.isPrimitive());
		System.out.println(int.class.isPrimitive());
		System.out.println(int.class == Integer.class);
		System.out.println(int.class == Integer.TYPE);
		System.out.println(int[].class.isPrimitive());
		System.out.println(int[].class.isArray());		
		
		//new String(new StringBuffer("abc"));
		Constructor constructor1 = String.class.getConstructor(StringBuffer.class);
		String str2 = (String)constructor1.newInstance(/*"abc"*/new StringBuffer("abc"));
		System.out.println(str2.charAt(2));
		
		ReflectPoint pt1 = new ReflectPoint(3,5);
		Field fieldY = pt1.getClass().getField("y");
		//fieldY的值是多少?是5,错!fieldY不是对象身上的变量,而是类上,要用它去取某个对象上对应的值
		System.out.println(fieldY.get(pt1));
		Field fieldX = pt1.getClass().getDeclaredField("x");
		fieldX.setAccessible(true);
		System.out.println(fieldX.get(pt1));	
		
		changeStringValue(pt1);
		System.out.println(pt1);
		
		Method methodCharAt = String.class.getMethod("charAt", int.class);
		System.out.println(methodCharAt.invoke(str1, 1));
		System.out.println(methodCharAt.invoke(str1, new Object[]{2}));
		
		//TestArguments.main(new String[]{"111","222","333"});
		String startingClassName = args[0];
		Method mainMethod = Class.forName(startingClassName).getMethod("main", String[].class);
		//mainMethod.invoke(null, new Object[]{new String[]{"111","222","333"}});
		mainMethod.invoke(null, (Object)new String[]{"111","222","333"});
		
		int [] a1 = new int[]{1,2,3};
		int [] a2 = new int[4];
		int[][] a3 = new int[2][3];
		String [] a4 = new String[]{"a","b","c"};
		System.out.println(a1.getClass() == a2.getClass());
		System.out.println(a1.getClass() == a4.getClass());
		System.out.println(a1.getClass() == a3.getClass());
		System.out.println(a1.getClass().getName());
		System.out.println(a1.getClass().getSuperclass().getName());
		System.out.println(a4.getClass().getSuperclass().getName());
		
		Object aObj1 = a1;
		Object aObj2 = a4;
		//Object[] aObj3 = a1;
		Object[] aObj4 = a3;
		Object[] aObj5 = a4;
		
		System.out.println(a1);
		System.out.println(a4);
		System.out.println(Arrays.asList(a1));
		System.out.println(Arrays.asList(a4));	
		
		printObject(a4);
		
		printObject("xyz");
	}

	private static void printObject(Object obj) {
		Class clazz = obj.getClass();
		if(clazz.isArray()){
			int len = Array.getLength(obj);
			for(int i=0;i<len;i++){
				System.out.println(Array.get(obj, i));
			}
		}else{
			System.out.println(obj);
		}
		
	}

	private static void changeStringValue(Object obj) throws Exception {
		Field[] fields = obj.getClass().getFields();
		for(Field field : fields){
			//if(field.getType().equals(String.class)){
			if(field.getType() == String.class){
				String oldValue = (String)field.get(obj);
				String newValue = oldValue.replace('b', 'a');
				field.set(obj, newValue);
			}
		}
		
	}

}


4、Constructor类(代表某个类中的一个构造方法)

(1)通过Class类获取实例对象,Constructor<T> getConstructor(Class<?>... parameterTypes);(2)创建实例对象,通过newInstance()方法,通常方式:String str = new String(new StringBuffer("abc"));,反射方式: String str = (String)constructor.newInstance(new StringBuffer("abc"));

5、Method类(代表某个类中的一个成员方法)

(1)通过Class类获取实例对象(2)调用得到的方法,Object invoke(Object obj, Object... args)

6、JavaBean(内省)(成员方法必须有getter和setter方法)

特点:JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则。
应用场景:如果要在两个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean的实例对象通常称之为值对象(Value Object,简称VO)。这些信息在类中用私有字段来存储,如果读取或设置这些字段的值,则需要通过一些相应的方法来访问。 
规则:JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。
PropertyDescriptor类(用于描述JavaBean):构造方法:PropertyDescriptor(String propertyName, Class<?> beanClass) ;获取get方法(只读):Method getReadMethod()  ;获取set方法(只写):Method getWriteMethod() 。
内省操作:简单的内省操作:(1)创建javabean对象(2)创建操作javabean类的PropertyDescriptor类的实例对象,并传递要操作的变量名(3)通过getReadMethod() 或getWriteMethod() 方法获取想要得到的方法(4)通过反射操作对方法进行调用等;
   复杂的内省操作:(1)创建javabean对象(2)通过Introspector类的静态方法static BeanInfo getBeanInfo(Class<?> beanClass, int flags)  获得javabean的所有属性和公开方法(3)通过BeanInfo的PropertyDescriptor[] getPropertyDescriptors()  方法获得一个PropertyDescripto数组(4)遍历数组,并用getName()方法与需要的属性名进行比较,拿到后通过PropertyDescriptor类获得get或set方法。(5)对拿到的方法进行反射操作,并break结束遍历。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值