Java基础21-反射

1.反射构造函数Constructor

package cn.base25.reflect;

import cn.base25.reflect.domain.Person;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

import org.junit.Test;

public class ReflectConstructor {

	/*
	 * 反射构造函数
	 * 		注: 1.使用getConstructor,只能反射public的构造方法
	 * 			2.使用getDeclaredConstructor,可以反射所有的构造函数
	 */
	@Test
	@SuppressWarnings({ "unchecked", "rawtypes", "unused" })
	public void test1() {
		
		try {
			//1-1.根据类名取得Class对象(Person.class),会将Persona.class加载进来
			Class clazz = Class.forName("cn.base25.reflect.domain.Person");
			
			/*			
			//创建Class对象的实例
			Person p = (Person) clazz.newInstance(); 				//注意: person类必须得有无参的构造函数,所以我们应该先反射其构造函数
			*/
			
			//1-2.反射构造函数,取到无参的构造函数
			Constructor constructor = clazz.getConstructor();		//根据实际构造函数是否有参数,来获取对应的构造函数
			
			//1-3.由Constructor对象,来创建Person对象
			Person p = (Person) constructor.newInstance();
			
			constructor = clazz.getConstructor(int.class);			//注意: 是int.class,千万别写成: Integer.class
			p = (Person) constructor.newInstance(20);
			
			//2.暴力反射private的构造函数
			constructor = clazz.getDeclaredConstructor(String.class, int.class);  	//使用的是: getDeclaredConstructor
			constructor.setAccessible(true);										//设为true,取消java语言访问检查
			p = (Person) constructor.newInstance("rose", 18);
			
			//3-1.列出Person类的所有构造函数(包括private)
			Constructor[] cons = clazz.getDeclaredConstructors();
			for(Constructor c : cons) {
				System.out.println(c.toString());
			}
			
			//3-2.使用getConstructor,只能反射public的构造方法
			System.out.println();
			Constructor[] cons2 = clazz.getConstructors();
			for(Constructor c : cons2) {
				System.out.println(c.toString());
			}
			
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (InstantiationException ex) {
			ex.printStackTrace();
		} catch (IllegalAccessException ex) {
			ex.printStackTrace();
		} catch (SecurityException e) {
			e.printStackTrace();
		} catch (NoSuchMethodException e) {
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			e.printStackTrace();
		}
	}
}

 

2.反射方法Method

package cn.base25.reflect;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import org.junit.Test;

import cn.base25.reflect.domain.Person;

public class ReflectMethod {

	/*
	 * 反射方法:
	 * 		注意: 1.clazz.getMethod(String name, Object...obj)--只会得到具体某一个方法,而不是同名的所有方法!!
	 * 			  2.使用getMethod,可以反射父类的方法!!!!!
	 */			  
	@SuppressWarnings({ "rawtypes", "unchecked" })
	@Test
	public void test1() {
		try {
			//0.加载Person类
			Class clazz = Class.forName("cn.base25.reflect.domain.Person");
			
			//1-1.使用getDeclaredMethod反射所有方法
			Method[] methods = clazz.getDeclaredMethods();		//使用getDeclaredMethod,会将Person类自身的所有方法列出(不包括父类的方法)
			for(Method m : methods) {
				System.out.println(m.toGenericString());
			}
			System.out.println();
			
			//1-2.使用getMethod反射所有方法
			methods = clazz.getMethods();						//使用getMethod,会将所有public的方法列出(包括父类继承而来的)
			for(Method m : methods) {
				System.out.println(m.toGenericString());
			}
			System.out.println();
			
			//2-1.取得一个无参的Person对象
			Person p = (Person) clazz.newInstance();
			
			//2-2.通过反射,调用方法
			Method m1 = clazz.getMethod("study");					//取得具体方法名的无参的method对象		
			//2-3.调用无参的study方法
			m1.invoke(p); 											//invoke方法的调用需要: 对象和实际参数
																	//这边也可以写成: m1.invoke(p, null)			
			//3-1.取得带一个int参数的"study"方法对象
			Method m2 = clazz.getMethod("study", int.class);    		
			//3-2.调用带int参数的study方法
			m2.invoke(p, 20);
			
			//4.反射private的方法--使用getDeclaredMethod,并setAccessible(true)
			Method m3 = clazz.getDeclaredMethod("study", int.class, String.class);
			m3.setAccessible(true);
			m3.invoke(p, 20, "sun");
			
			//5.反射静态方法
			Method m4 = clazz.getMethod("study", String.class);
			m4.invoke(p, "sta");									//这边也可写成: m4.invoke(null, "sta");
			
			//6.反射main主函数
			Method m5 = clazz.getMethod("main", String[].class);	//注意: String[].class
			m5.invoke(null, (Object)new String[]{"test"});			//这边也可以写成: m5.invoke(null, new Object[]{new String[]{"test2"}});
			
			//7.反射Object的方法										//注意: 根据上面可以反射父类的方法知,可以反射那些方法
			Method m6 = clazz.getMethod("hashCode");
			System.out.println(m6.invoke(p));
			
			
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (SecurityException e2) {
			e2.printStackTrace();
		} catch (NoSuchMethodException e2) {
			e2.printStackTrace();
		} catch (IllegalArgumentException e3) {
			e3.printStackTrace();
		} catch (IllegalAccessException e3) {
			e3.printStackTrace();
		} catch (InvocationTargetException e3) {
			e3.printStackTrace();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

3.反射字段Field

package cn.base25.reflect;

import java.lang.reflect.Field;

import org.junit.Test;

import cn.base25.reflect.domain.Student;

public class ReflectField {

	/*
	 * 反射字段:
	 * 		注意: 1.使用getDeclaredFields,只会列出Student自身的字段(而不包括Person的),不能反射Person的字段!!!
	 * 			 2.一个Method/Field对象,对应一个具体方法/一个具体字段,但同时可以对应多个类对象的这一个方法/字段!!!
	 * 			3.判断出字段的类型!!!!
	 */
	@SuppressWarnings({ "rawtypes" })
	@Test
	public void test1() {
		
		try {
			//1.加载Student类
			Class clazz = Class.forName("cn.base25.reflect.domain.Student");
			
			//2-1.使用getDeclaredFileds,反射出所有的字段		//使用getDeclaredFields,只会列出Student自身的字段(而不包括Person的)
			Field[] fields = clazz.getDeclaredFields();
			for(Field f : fields) {
				System.out.println(f.toGenericString());
			}
			System.out.println();
					
			//2-2.使用getFileds,反射出所有的字段
			Field[] fields2 = clazz.getFields();			//使用getMethods,只会列出public的Student字段
			for(Field f : fields2) {
				System.out.println(f.toGenericString());
			}
			
			//3.创建一个Student对象
			Student stu = (Student) clazz.newInstance();
			
			//4.设置public的"sName"字段的值,并取出,查看结果
			Field field = clazz.getField("sName");
			field.set(stu, "李四");
			System.out.println(field.get(stu)); 			//打印stu对象的sName的值
			
			//5.反射private的"name3"字段
			Field field2 = clazz.getDeclaredField("name3");	//1.需要使用getDeclaredField
			field2.setAccessible(true);						//2.还要打开权限
			field2.set(stu, "王伟");
			System.out.println(field2.get(stu));
			
			//6.取得"name3"字段的类型,判断出其类型
			Class cType = field2.getType();
			if(String.class.equals(cType)) {
				String s  = (String) field2.get(stu);
				System.out.println("\"name3\"字段的类型是 " + s.getClass() +", 值为:" + s);
			}
			
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (SecurityException e) {
			e.printStackTrace();
		} catch (NoSuchFieldException e) {
			e.printStackTrace();
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (InstantiationException e) {
			e.printStackTrace();
		}
	}
}


附:

1.Person类:

package cn.base25.reflect.domain;

@SuppressWarnings("unused")
public class Person {
	public String name="张三";
	public int age;
	
	//构造函数
	public Person() {
		System.out.println("person()");
	}
	
	public Person(int id) {
		System.out.println("public person(int)");
	}
	
	private Person(String name, int id) {
		System.out.println("private person(String, int)");
	}
	
	Person(int id, int j, int m) {
		System.out.println("public person(int)");
	}
	
	protected Person(int id, int j) {
		System.out.println("public person(int)");
	}
	
	//方法
	public void study() {
		System.out.println("public study()");
	}	
	
	void study(int i, int j, String s) {
		System.out.println("默认 study(int i, int j, String s)");
	}
	
	public void study(int i) {
		System.out.println("public study(int i)");
	}	
	
	protected void study(int i, int j) {
		System.out.println("protected study(int i, int j)");
	}
	
	private void study(int i ,String s) {
		System.out.println("private study(int i ,String s)");
	}
	
	public static void study(String s) {
		System.out.println("public static study(String s)");
	}
	
	public static void main(String []args) {
		System.out.println("reflect main() sucessful!");		
	}
}

 

2.Student类:

package cn.base25.reflect.domain;

@SuppressWarnings("unused")
public class Student {
	public String sName;
	String name1;
	protected String name2;
	private String name3;
	public int age;
	public static int id;
	
	//构造函数
	public Student() {
		System.out.println("person()");
	}
	
	public Student(int id) {
		System.out.println("public person(int)");
	}
	
	private Student(String name, int id) {
		System.out.println("private person(String, int)");
	}
	
	Student(int id, int j, int m) {
		System.out.println("public person(int)");
	}
	
	protected Student(int id, int j) {
		System.out.println("public person(int)");
	}
	
	//方法
	public void study() {
		System.out.println("public study()");
	}	
	
	void study(int i, int j, String s) {
		System.out.println("default study(int i, int j, String s)");
	}
	
	public void study(int i) {
		System.out.println("public study(int i)");
	}	
	
	protected void study(int i, int j) {
		System.out.println("protected study(int i, int j)");
	}
	
	private void study(int i ,String s) {
		System.out.println("private study(int i ,String s)");
	}
	
	public static void study(String s) {
		System.out.println("public static study(String s)");
	}
	
	public static void main(String []args) {
		System.out.println("reflect main() sucessful!");		
	}
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章