0034 Java学习笔记-反射-初步2-操作对象

通过反射创建对象

  • 通过反射创建对象有两种方式,一种通过Class对象的newInstance()方法,一种是获取到Class对象的Constructor后,再调用newInstance()方法,前者要求对应类有默认的无参构造器,后者则可以用特定的的构造器
  • 前者用得较多,很多Java EE框架中都用xml配置文件信息来创建Java对象
  • 示例:
package testpack;

import java.lang.reflect.Constructor;

public class Test1  { 
    public static void main(String[] args)throws Exception{ 
    	Class clazz=A.class;
    	Object obj1=clazz.newInstance();           //调用Class对象的newInstance()方法,用默认构造器创建
    	System.out.println(obj1);                  //输出:[name= default , age= 0 ]
    	
    	Constructor c=clazz.getConstructor(String.class,int.class);
    	Object obj2=c.newInstance("Java",18);      //调用Constructor的newInstance()方法用特定构造器创建
    	System.out.println(obj2);                  //输出:[name= Java , age= 18 ]
    }
}
class A{
	private String name;
	public int age;
	A(){
		name="default";
		age=0;
	}
	public A(String name,int age){
		this.name=name;
		this.age=age;
	}
	public String toString(){
		return "[name= "+name+" , age= "+age+" ]";
	}
}

通过反射调用方法

  • 获得一个Class对象的Method对象后,通过调用invoke(Object obj,Object...args)方法就可以调用特定对象的该方法了,obj就是该特定对象,args是要传入的实参
  • 见示例:
package testpack;

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

public class Test1  { 
    public static void main(String[] args)throws Exception{ 
    	Class clazz=A.class;                         //获取类A的Class对象
    	Constructor c=clazz.getConstructor(String.class,int.class); //获取构造方法
    	Object obj=c.newInstance("C++",28);          //通过反射创建对象
    	Method m=clazz.getMethod("show",null);       //获取名为show,无参的方法
    	m.invoke(obj,null);                          //在obj上调用show方法,传入参数为null,就无实参
    }
}
class A{
	private String name;
	public int age;
	A(){
		name="default";
		age=0;
	}
	public A(String name,int age){
		this.name=name;
		this.age=age;
	}
	public String toString(){
		return "[name= "+name+" , age= "+age+" ]";
	}
	public void show(){
		System.out.println("现在在调用show()方法,我是:"+toString());
	}
}

通过反射访问成员变量

  • 获取成员变量的值用get(Objcet obj)、getInt(Object obj)、getLong(Object obj)、...以此类推
  • 设置成员变量的值用set(Objcet obj)、setInt(Objcet obj)、setLong(Objcet obj)、...以此类推
  • 如果是private修饰的变量,在访问之前用setAccessible(true)取消访问权限检查
  • 见示例:
package testpack;

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

public class Test1  { 
    public static void main(String[] args)throws Exception{ 
    	Class clazz=A.class;
    	Constructor c=clazz.getConstructor(String.class,int.class);
    	Object obj=c.newInstance("C++",28);
    	System.out.println(obj);                         //输出:[name= C++ , age= 28 ]

    	Field nameField=clazz.getDeclaredField("name");  //获得name这个private成员变量
    	nameField.setAccessible(true);                   //取消对该变量的访问权限检查
    	System.out.println("private name="+nameField.get(obj));  //输出:private name=C++
    	Field ageField=clazz.getField("age");            //获得age这个public成员变量
    	System.out.println("public age= "+ageField.getInt(obj)); //输出:public age= 28
    	
    	nameField.set(obj,"Java");                       //将obj的private的name变量改为“Java”
    	ageField.setInt(obj, 100);                       //将obj的public的age变量改为100
    	System.out.println("修改了对象的private和public实例变量:"+obj); //输出:修改了对象的private和public实例变量:[name= Java , age= 100 ]
    }
}
class A{
	private String name;
	public int age;
	A(){
		name="default";
		age=0;
	}
	public A(String name,int age){
		this.name=name;
		this.age=age;
	}
	public String toString(){
		return "[name= "+name+" , age= "+age+" ]";
	}
	
}

  • setAccessible()方法是Constructor、Field、Method的父类的AccessibleObject的方法,因此这三个类都可以用
  • 见示例:
package testpack;

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

public class Test1  { 
    public static void main(String[] args)throws Exception{ 
    	Class clazz=A.class;
    	Constructor c=clazz.getDeclaredConstructor(String.class,int.class);
    	c.setAccessible(true);                           //取消该构造器的访问权限检查
    	Object obj=c.newInstance("Java",18);             //调用私有构造器
    	System.out.println(obj);                         
    	
    	Method m=clazz.getDeclaredMethod("show",null);   
    	m.setAccessible(true);                           //取消show()方法的访问权限检查
    	m.invoke(obj, null);                             //调用私有方法
    }
}
class A{
	private String name;
	public int age;
	A(){
		name="default";
		age=0;
	}
	private A(String name,int age){                       //private构造器
		this.name=name;
		this.age=age;
		System.out.println("调用了private构造器");
	}
	public String toString(){
		return "[name= "+name+" , age= "+age+" ]";
	}
	private void show(){                                  //private方法
		System.out.println("在调用private方法:"+toString());
	}
}

通过反射操作数组

  • java.lang.reflect.Array可用来操作数组
  • 创建一个数组:
    • static Object newInstance(Class<?>componentType,int...length):两个参数一个是类型,一个是长度
  • 访问某个数组元素:
    • static xxx getXxx(Object array,int index):返回array数组的第index个元素;引用类型则是:get(Object array,int index)
  • 设置某个数组元素的值
    • static void setXxx(Object array,int index,xxx value):设置数组array元素的第index个元素的值为value,引用类型:set(....)
  • 获取数组的长度
    • static int getLength(Object array):获得数组array的长度
  • 见示例:
package testpack;
import java.lang.reflect.Array;
public class Test1  { 
    public static void main(String[] args)throws Exception{ 
    	Object arr=Array.newInstance(String.class, 5);  //调用构造方法
    	Array.set(arr, 0, "Java");                      //将索引号为0的元素设置为“Java”
    	Array.set(arr, 1, "C++");
    	Array.set(arr, 2, "python");
    	System.out.println(Array.get(arr, 2));          //取得索引号为2的元素
    	System.out.println("array的长度:"+Array.getLength(arr)); //输出:array的长度:5
    }
}
posted @ 2016-11-30 17:27  sonng  阅读(353)  评论(0编辑  收藏  举报