码迷,mamicode.com
首页 > 其他好文 > 详细

反射技术总结

时间:2015-03-20 14:32:38      阅读:155      评论:0      收藏:0      [点我收藏+]

标签:

反射技术;其实就是动态加载一个指定的类,并获取该类中的所有内容。
而且将字节码文件与内容都封装成了对象。
这样便于操作这些成员,简单说;反射技术可以对一个类进行解刨。
反射的好处;大大的增强了程序的扩展性。
反射的基本步骤;
获得Class对象。就是获取到指定的名称的字节码文件对象。
实例化对象,获得类的属性,方法或构造函数。
访问属性,调用方法,调用构造函数创建对象。
获取Class对象有三种方法。
每个类都具备getClass();方法,弊端;必须要创建该类对象,才可以调用该方法。
每一个数据类型(基本数据类型和引用数据类型)都有一个静态的属性class。
前两种方式不利于程序的扩展,因为都需要在程序中使用具体的类来完成。
使用Class类中的静态方法forName。
指定什么类名,就获取什么类的字节码文件对象,输入类名字符串即可,扩展性强。
有八个基本数据类型。每一个基本数据类型都对应着Class字节码文件对象,void.class等等--
import java.lang.reflect.*;
class most1{
	public static void main(String[] args)throws ClassNotFoundException{
		String s1="abc";
		Class c1=s1.getClass();
		Class c2=String.class;
		Class c3=Class.forName("java.lang.String");
		System.out.println(c1==c2);
		System.out.println(c1==c3);
		System.out.println(c1.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());
	}
}
反射;就是把java类中的各个成分映射成相应的类,反射比较消耗资源。
得到类的所有构造方法。
Constructor[] tts=Class.forName("java.lang.String").getConstructors();
得到类的一个构造方法。
Constructor tts=Class.forName("java.lang.String").getConstructor(StringBuffer.class);
import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args)throws NoSuchMethodException,InstantiationException,IllegalAccessException,InvocationTargetException{
		Constructor tts=String.class.getConstructor(StringBuffer.class);
		String s1=(String)tts.newInstance(new StringBuffer("abc"));
		System.out.println(s1.charAt(2));
	}
}
获取类的变量:ReflectPoint pt=new ReflectPoint(3,5);
Field fieldY=pt.getClass().getField("y");
System.out.println(field.get(pt));
获取类的私有变量:Field fieldX=pt.getClass().getDeclaredField("x");
fieldX.setAccessible(true);
System.out.println(fieldX.get(pt));
import java.lang.reflect.*;
class ReflectPoint{
	public int x;
	private int y;
	ReflectPoint(int x,int y){
		this.x=x;
		this.y=y;
	}
}
class ReflectDemo{
	public static void main(String[] args)throws NoSuchFieldException,IllegalAccessException{
		ReflectPoint pt=new ReflectPoint(3,3);
		Field fieldX=pt.getClass().getField("x");
		System.out.println(fieldX.get(pt));
		Field fieldY=pt.getClass().getDeclaredField("y");
		fieldY.setAccessible(true);
		System.out.println(fieldX.get(pt));
	}
}
获取类的方法;// Class c1=obj.getClass();
Method method=Object.class.getMethod("charAt",int.class);
System.out.println(method.invoke(obj,index));
import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args)throws NoSuchMethodException,IllegalAccessException,InvocationTargetException{
		String s="abc";
		Class c1=s.getClass();
		Method method=String.class.getMethod("charAt",int.class);
		System.out.println(method.invoke(s,1));
		System.out.println(method.invoke(s,new Object[]{2}));
	}
}
反射的用法;首先需要提供类的Class对象,获得Class对象的三种方式。
Class.forName() 用于做类加载。
obj.getClass() 用于获得对象的类型。
类名.class() 用于获得指定的类型,传参用。
获取了字节码文件对象后,最终都需要创建指定类的对象。
创建对象的两种方式(其实就是对象在进行实例化时的初始化方式)
调用空参数的构造函数;使用了Class类中的newInstance方法。
调用带参数的构造函数;先要获取指定参数列表的构造函数对象。
然后通过该构造函数对象的newInstance进行对象的初始化。
替换一个类的变量内容;
import java.lang.reflect.*;
class ReflectPoint{
	String s1="abll";
	String s2="base";
	String s3="itcast";
	public String toString(){
		return s1+"--"+s2+"--"+s3;
	}
}
class ReflectDemo{
	public static void main(String[] args)throws IllegalAccessException{
		ReflectPoint pt=new ReflectPoint();
		Field[] fields=pt.getClass().getDeclaredFields();
		for(Field field:fields){
			if(field.getType()==String.class){
				String str=(String)field.get(pt);
				String value=str.replace('b','a');
				field.set(pt,value);
			}
		}
		System.out.println(pt);
	}
}
写一个程序;这个程序能够根据用户提供的类名去执行该类中的main方法。
启动java程序的main方法是一个字符数组,public static void main(String[] args){}
通过反射方式来调用这个main方法,按照jdk1.5的语法给invoke传递参数。
整个数组就是一个参数,而jdk1.4的语法数组中的每一个元素都对应一个参数。
当把一个字符串数组作为参数传递给invoke方法时,1.5兼容1.4的语法。
会按照1.4的语法进行处理,就将数组打散成若干个元素。
所以在给main方法传递参数时,不能使用代码。
method.invoke(null,new String[]{"","",""});java只把他当做1.4的语法处理。
而不把他当做1.5的语法处理,因此会出现参数类型不对的问题。
method.invoke(null,new Object[]{new String[]{"","",""}});
method.invoke(null,(Object)new String{}{"","",""});
编译器会作特殊处理;编译时不把参数当做数组看待,也就不会将数组打散。
import java.lang.reflect.*;
class ReflectPoint{
	public static void main(String[] args)throws InvocationTargetException,ClassNotFoundException,NoSuchMethodException,IllegalAccessException{
		for(String str: args){
			System.out.println(str);
		}
	}
}
class ReflectDemo{
	public static void main(String[] args)throws InvocationTargetException,ClassNotFoundException,NoSuchMethodException,IllegalAccessException{
		String startName=args[0];
		Method mainMethod=Class.forName(startName).getMethod("main",String[].class);
		mainMethod.invoke(null,(Object)new String[]{"111","222","333"});
	}
}
具有相同维数和元素类型的数组属于同一类型,具有相同Class实例对象。
代表数组的Class实例化对象的getSuperclass方法返回的是父类对应的Class。
基本类型的一维数组可以被当做Object类型使用,不能当做Object[]类型使用。
非基本类型的一维数组,既可以当做Object类型使用,又可以当做Object[]类型使用。
Arrays.asList方法处理int[]和String[]时的差异。
Array工具类用于完成数组的反射操作。
思考题;怎么得到数组中的元素类型?
import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args){
		String[] arr={"x","y","z"};
		printObject(arr);
		printObject("xyz");
	}
	public static void printObject(Object obj){
		Class c=obj.getClass();
		if(c.isArray()){
			int len=Array.getLength(obj);
			for(int x=0; x<len; x++){
				System.out.println(Array.get(obj,x));
			}
		}else{
			System.out.println(obj);
		}
	}
}
package pack;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
class ReflectPoint{
<span style="white-space: pre;">	</span>private int x;
<span style="white-space: pre;">	</span>private int y;
<span style="white-space: pre;">	</span>ReflectPoint(int x,int y){
<span style="white-space: pre;">		</span>this.x=x;
<span style="white-space: pre;">		</span>this.y=y;
<span style="white-space: pre;">	</span>}
<span style="white-space: pre;">	</span>public boolean equals(Object obj){
<span style="white-space: pre;">		</span>if(!(obj instanceof ReflectPoint))
<span style="white-space: pre;">			</span>throw new RuntimeException("类型不匹配");
<span style="white-space: pre;">		</span>ReflectPoint r=(ReflectPoint)obj;
<span style="white-space: pre;">		</span>return this.x==(r.x)&&this.y==r.y;
<span style="white-space: pre;">	</span>}
}
class ReflectDemo{
<span style="white-space: pre;">	</span>public static void main(String[] args){
<span style="white-space: pre;">		</span>Collection coll=new ArrayList();
<span style="white-space: pre;">		</span>ReflectPoint pt1=new ReflectPoint(3,3);
<span style="white-space: pre;">		</span>ReflectPoint pt2=new ReflectPoint(5,5);
<span style="white-space: pre;">		</span>ReflectPoint pt3=new ReflectPoint(3,3);
<span style="white-space: pre;">		</span>coll.add(pt1);
<span style="white-space: pre;">		</span>coll.add(pt2);
<span style="white-space: pre;">		</span>coll.add(pt3);
<span style="white-space: pre;">		</span>coll.add(pt1);
<span style="white-space: pre;">		</span>System.out.println(coll.size());
<span style="white-space: pre;">	</span>}
}

反射技术总结

标签:

原文地址:http://blog.csdn.net/qiang3570/article/details/44489517

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!