java反射

                                                                          Java反射机制

         

             大家都知道,java不是一个动态语言,但是java不像c、c++的纯静态语言一样,它有个Reflection类能让它在运行的过程中获取相关类的信息,这就是java反射机制。

            java反射通俗的说就是程序在运行的过程中可以获得某个类的全部信息。

            而说到反射,就不能不说一下Class类,每一个类都会有一个Class对象,在编译生成的.Class文件中,就会产生一个class对象,来表示类型信息,我们就可以通过反射来调用之。用法比较简单,直接上代码。

           我建立了一个Dog的测试类,它有各种声明类型不同的属性以及限制符不同的方法    

package february24;
public class Dog {
	//定义dog类中的一些属性和方法。
	public String name;
	private int weight;
	protected int height;
	public int age = 5;
	//构造函数
	public Dog(){
	}
	private Dog(int age){
		this.age=age;
	}
	public Dog(String name){
		this.name=name;
	}
	public void bark(){
		System.out.println("汪汪.....");
	}
	private void bark0(){
		System.out.println("汪汪.....");
	}
	public void bark(String name){
		System.out.println(name+"汪汪汪。。。");
	}
}

       

              1:反射可以根据某个对象而得到其类的出处。

	public static void test0(){
		try{
			Dog jack = new Dog();
			String cName = jack.getClass().getName();
			System.out.println("jack对象所属类的路径为:"+cName);
		}
		catch(Exception e){
			e.printStackTrace();
		}
	}

 

 运行结果为:jack对象所属类的路径为:february24.Dog

                 2:反射可以得到类的方法信息。

// 反射在知道类的名字后,可以得倒某个类的所有方法。
	public static void test1() {
		try {
			Class c = Class.forName("february24.Dog");
			//getMethods方法可以获得类的公开声明的成员方法以及其继承和实现的类(接口)的公开声明的成员方法
			Method mlist[] = c.getMethods();
			for (int i = 0; i < mlist.length; i++) {
				System.out.println("Dog中有一个" + mlist[i].toString() + "方法****");
			}
			//getDeclaredMethods只能得到本类的方法(即使是私有声明的)
			Method mlists[] = c.getDeclaredMethods();
			for (int i = 0; i < mlists.length; i++) {
				System.out.println("Dog中有一个" + mlists[i].toString() + "方法++++");
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

                 其中,getMethods()方法可以得到某个类以及其父类或其实现的接口的公开声明的方法。getDeclaredMethod()方法能得到某个类中声明的所有方法。上面代码的结果为:

Dog中有一个public void february24.Dog.bark(java.lang.String)方法****
Dog中有一个public void february24.Dog.bark()方法****
Dog中有一个public final native java.lang.Class java.lang.Object.getClass()方法****
Dog中有一个public native int java.lang.Object.hashCode()方法****
Dog中有一个public boolean java.lang.Object.equals(java.lang.Object)方法****
Dog中有一个public java.lang.String java.lang.Object.toString()方法****
Dog中有一个public final native void java.lang.Object.notify()方法****
Dog中有一个public final native void java.lang.Object.notifyAll()方法****
Dog中有一个public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException方法****
Dog中有一个public final void java.lang.Object.wait() throws java.lang.InterruptedException方法****
Dog中有一个public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException方法****
Dog中有一个public void february24.Dog.bark(java.lang.String)方法++++
Dog中有一个public void february24.Dog.bark()方法++++
Dog中有一个private void february24.Dog.bark0()方法++++

                      3:与上面通过反射得到类的方法类似,通过反射可以得到类的构造函数和成员属性

得到构造函数:

//得到类中的构造函数
	public static void test2(){
		try{
			Class c = Class.forName("february24.Dog");
			Constructor conlist [] = c.getConstructors();
			//得到类中公开的构造函数
			for(int i = 0;i<conlist.length ;i++){
				System.out.println("Dog类中的构造函数为:"+conlist[i].toString()+"****");
			}
			//可以得到所有的类中所有的构造方法
			Constructor conlists [] = c.getDeclaredConstructors();
			for(int i = 0;i<conlists.length;i++){
				System.out.println("Dog类中的构造函数为:"+conlists[i].toString()+"++++");
			}
		}
		catch(Exception e){
			e.printStackTrace();
		}
	}

                   运行结果为:

Dog类中的构造函数为:public february24.Dog()****
Dog类中的构造函数为:public february24.Dog(java.lang.String)****
Dog类中的构造函数为:public february24.Dog()++++
Dog类中的构造函数为:private february24.Dog(int)++++
Dog类中的构造函数为:public february24.Dog(java.lang.String)++++

 得到成员属性:

//反射可以得到某个类中的定义的属性。
	public static void test3(){
		try{
			Class c = Class.forName("february24.Dog");
			Field flist[] = c.getFields();
			//这个方法只能得到Dog中的public方法。protected,private以及default均得不到。
			for(int i =0;i<flist.length;i++){
				System.out.println("Dog中有"+flist[i].toString()+"属性****");
			}
			Field flists[] = c.getDeclaredFields();
			for(int i =0;i<flists.length;i++){
				System.out.println("Dog中有"+flists[i].toString()+"属性++++");
			}
		}
		catch (Exception e){
			e.printStackTrace();
		}
	}

 其运行结果为:

 

Dog中有public java.lang.String february24.Dog.name属性****
Dog中有public int february24.Dog.age属性****
Dog中有public java.lang.String february24.Dog.name属性++++
Dog中有private int february24.Dog.weight属性++++
Dog中有protected int february24.Dog.height属性++++
Dog中有public int february24.Dog.age属性++++
 

 

                           4:利用反射还可以实现类似于instanceof功能。

	//反射可以有着类似于instanceof的功能。
	public static void test4(){
		try {
			Dog jack = new Dog();
			String a = new String("faith");
			Class c = Class.forName("february24.Dog");
			boolean b0 = jack instanceof Dog;
			boolean b1 =c.isInstance(a);
			System.out.println(b0+"$$$"+b1);
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
	}

                            输出结果为:

true$$$false

                         虽然做这种判断因为没有利用好java多态的优点而不推荐使用,但也算是一种java反射的用处。

                       5:利用发射可以创建新的对象以及调用类中的方法

public static void test5(){
		try{
		//首先获取类
		Class c = Class.forName("february24.Dog");
		Method m = c.getMethod("bark");
		m.invoke(c.newInstance());//其中c.newInstance()是新构造一个对象出来。
		}
		catch(Exception e){
			e.printStackTrace();
		}
	}
public static void test6(){
		try {
			Class c = Class.forName("february24.Dog");
			Method m = c.getMethod("bark",String.class);//具体调用方法的参数不一样,调用的方法不同
			m.invoke(c.newInstance(),"小黑:");//构造新对象,同时确定参数。
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

 得到的结果为:

汪汪.....
小黑:汪汪汪。。。

                  Class中有一个newInstance()方法可以创建新的对象。同时,我们也可以在程序运行的时候通过给定的参数利用Construction来创建新的对象,而不是在其编译的时候创建。

	public static void test7(){
		try {
			Class c = Class.forName("february24.Dog");
			Constructor con = c.getConstructor(String.class);
			Field f = c.getField("name");
			System.out.println(f.get(con.newInstance("Jack")));
		} catch (Exception e) {
			e.printStackTrace();
		}
	}	

 输出的结果为:

Jack

 

                             6:利用反射修改类中的属性值

public static void test8(){
		try {
			Class c = Class.forName("february24.Dog");
			Object obj = c.newInstance();
			Field f = c.getField("age");
			f.setAccessible(true);//这个使得它能够修改变量
			f.set(obj,10);
			System.out.println("小狗的年龄为"+f.get(obj));
			
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

                  结果为:

小狗的年龄为10

            在刚开始测试的时候,我没有定义一个Object obj,在上面代码中的obj全是一个c.newInstance(),结果输出值为5。。。这个错误真是有些低级啊。。。

             反射还有一个Modifier类可以得倒某个方法或属性的修饰符。

public static void test9(){
		try {
			Class c = Class.forName("february24.Dog");
			Field f[] = c.getDeclaredFields();
			for(int i=0;i<f.length;i++){
				int a =f[i].getModifiers(); 
				System.out.println(f[i].toString()+"的限制符为:"+Modifier.toString(a));
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

 结果为:

public java.lang.String february24.Dog.name的限制符为:public
public int february24.Dog.age的限制符为:public
private int february24.Dog.weight的限制符为:private
protected int february24.Dog.height的限制符为:protected      

                         以上内容就是我对反射机制的基本尝试,在测试的过程中,我发现从点击运行到出来结果,虽然是一个很小的程序段,但是却不能立即得出答案。。。估计反射的应用会降低性能。同时,有什么不对的地方,欢迎大家指正。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值