黑马程序员_Java高新技术_反射

------- android培训java培训、期待与您交流! ----------
反射不是java 1.5的新特性,以前的版本中就有的特性,在java 开发中常见的框架框架Struts,Spring ,Hibernate等框架的设计都用到反射的技术。
Class 类(反射的基础)
java 类用于描述一类事物的共性,该类事物有什么属性,没有什么属性,至于这个属性的值是什么,是由这个类的实例对象来确定的,不同的实例对象有不同的属性值,Java 程序中的各个类属于同一类事物,描述这类事物的类名就是Class。
例如,众多的人用Person 类表示,众多的java类用Class 表示
Person 类代表人,它的实例对象对应一个个具体的人,Class 代表java类它的各个实例对象对应着什么?
Class 类的 实例对象对应着各个类在内存中的字节码。
一个类被加载器加载到内存中,占用一片存储空间,这个存储空间里的内容就是字节码,不同的类的字节码是不同的。

获取字节码对应的实例对象的三种方式(Class 型)
1,类名.class 例如 System.class 
2,对象.getClass() 例如 new Date().getClass();
3,Class.forName(类名) 例如Class.forName("java.util.Date");//反射用这种
9个预定义的class实例对象:
8个基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。 
注意
int.class==Integer.TYPE
总之只要在源程序中出现的类型,都有各自的class 实例对象,如,int[]。void
理解反射
反射就是把类中的各种成分映射成相应的java类。例如一个java类用Class 类的对象来表示,一个类中的组成部分,成员变量,构造方法方法,包等信息也用
一个个的java类表示。就像汽车是一个类,汽车中的发动机,变速箱等也是一个类。表示java类的Class 类显然要提供一系列的方法,显然要提供一系列的方
法,来获得类中的构造方法,方法,修饰符,变量,包等信息,这些信息就是用类的相应类的实例对象表示,他们是Field,Method,Contructor,Package
构造方法的反射应用
Constructor 代表某个类中的构造方法
得到某个类的所有的构造方法
Constructor[] constructors=Class.forName("java.lang.String").getConstructors();
得到某个构造方法
Constructor constructor=Class.forName("java.lang.String").getConstructor(StringBuffer.class);//获得方法时要用到类型,参数的类型
反射方式获得实例对象
Constructor constructor=String.class.getConstructor(StringBuffer.class);
String str=(String)constructor.newInstance(new StringBuffer("abc"));
调用获得的方法时要用到上面相同类型的实例对象
Class.newInstance()
该方法内部先得到默认的无参数的构造方法,然后由该构造方法创建实例对象
例子 String ss=Class.forName("java.lang.String").newInstance();
成员变量的反射
Field 类代表某个类的成员变量,得到的Field对象是对应到类上面的成员变量,不是对应到对象的成员变量
通过变量名获得对象的 字节码再获得字段对象
Field fieldY=pt1.getClass().getField("y");
暴力反射:当变量是私有或者不可见的,可以通过getDeclaredField方法获取到字段,再通过setAccessible(true),实现对字段的访问
Field fieldX=pt1.getClass().getDeclaredField("x");//x是私有的不可见的用该方法获得field对象
//暴力反射
fieldX.setAccessible(true);//因为x是私有的不可以直接访问,设置为可访问的
成员方法的反射
Method 类代表某个类中的成员方法
得到类中的某个方法 :
Method methodCharAt=String.class.getMethod("charAt", int.class);
//调用方法通常方式str1.charAt(1);
反射方式调用成员方法:
System.out.println(methodCharAt.invoke(s, 1));
如果传递给Method 的第一个参数为null,说明Method对应的是一个静态的方法
JDK1.4和JDK1.5 invoke()的区别
JDK1.5 添加了可变参数JDK 1.4,参数为一个数组
public Object invoke(Object obj, Object...args);
public Object invoke(Object obj, Object[]args);

用反射的方式执行某个类中的main方法
目标:一个程序,这个程序能够根据用户提供的类名,去调用该类的main 方法
问:为什么要用反射的方式调用main方法
启动java程序main方法的参数是一个String类型的数组,即public static void main(String [] args)
通过反射方式调用main方法时,如何为invoke方法传递参数?
JDK1.5 整个数组就是一个参数,jdk1.4数组中的每个元素对应一个参数,当把字符串数组作为参数传递给invoke时,由于JDK1.5要兼容1.4,会按照1.4 的语法进行
处理,即把数组打散成若干个单独的参数,所以给main方法传递参数时不能用mainMethod.invoke(null,new String("xxx")),
会报错,要按照1.4版本方法
String startingClassName=args[0];
		//用反射调用类Arguments 的main方法,获取Method 类型的main 方法的反射
		Method mainMethod=Class.forName(startingClassName).getMethod("main", String[].class);
		//按照jdk1.4取。方式一:参数为object数组类型,静态方法不需要传递对象!!
		mainMethod.invoke(null,new Object[]{new String[]{"aaa","bbb","ccc"}} );
		//方式二:参数为强转成Object 类型的string 数组,因为String 数组会被1.4 看做一个对象参数,
		//而1.4 接收的是一个object数组,静态方法不需要传对象
		mainMethod.invoke(null,(Object)new String[]{"pp","qq","rr"});
数组的反射
具有相同维数和元素类型的数组属于同一个类型,即具有相同的Class实例对象(字节码相同)
代表数组的Class 实例对象的getSuper()方法返回的父类是Object类对应的Class
基本类型的一维数组可以被当做Object 类型使用,不能当做不能当做Object[]类型使用(因为基本数据类型不属于Object)
非基本类型的一维数组既可以当做Object类型也可以当做Object[] 类型使用(因为String是Object 类型)
这就是Arrays.asList()处理int[] 和String []的差异的原因。
int[] a1=new int[]{2,3,4};
		int[] a2=new int [4];
		int[][] a3=new int [2][3];
		String[] a4= new String[]{"aa","bb","cc"};
		System.out.println(a1.getClass()==a2.getClass());
		System.out.println(a1.getClass().getName());
		//基本数据类型的数组和非基本数据类型的数组的父类都是Object
		System.out.println(a1.getClass().getSuperclass().getName());
		System.out.println(a4.getClass().getSuperclass().getName() );
		Object obj1=a1;
		Object Obj4=a4;
		//Object[] obj3=a1;int 不是Object不可以
		Object [] obj4=a4;
		//获取数组中的元素
		System.out.println(Arrays.asList(a1));
		System.out.println(Arrays.asList(a4));
数组的反射应用
Array 工具类用于完成对数组反射的操作
private static void printObject(Object obj) {
		if(obj.getClass().isArray())
		{//数组的Class 类型是Array
			int len=Array.getLength(obj);
			for(int i=0;i<len;i++)
			{
				System.out.println(Array.get(obj, i));
			}
		}
		else
			System.out.println(obj);
		
	}


------- android培训java培训、期待与您交流! ----------
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值