Java在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。
[TOC]
反射机制的相关类
与Java反射相关的类如下:
类名 | 用途 |
Class类 | 代表类的实体,在运行的Java应用程序中表示类和接口 |
Field类 | 代表类的成员变量(成员变量也称为类的属性) |
Method类 | 代表类的方法 |
Constructor类 | 代表类的构造方法 |
Class代表类的实体,在运行的Java应用程序中表示类和接口。在这个类中提供了很多有用的方法,这里对他们简单的分类介绍。
方法 | 用途 | |
asSubclass(Class<U> clazz) | 把传递的类的对象转换成代表其子类的对象 | |
Cast | 把对象转换成代表类或是接口的对象 | |
getClassLoader() | 获得类的加载器 | |
getClasses() | 返回一个数组,数组中包含该类中所有公共类和接口类的对象 | |
getDeclaredClasses() | 返回一个数组,数组中包含该类中所有类和接口类的对象 | |
forName(String className) | 根据类名返回类的对象 | |
getName() | 获得类的完整路径名字 | |
newInstance() | 创建类的实例 | |
getPackage() | 获得类的包 | |
getSimpleName() | 获得类的名字 | |
getSuperclass() | 获得当前类继承的父类的名字 | |
getInterfaces() | 获得当前类实现的类或是接口 |
方法 | 用途 |
getField(String name) | 获得某个公有的属性对象 |
getFields() | 获得所有公有的属性对象 |
getDeclaredField(String name) | 获得某个属性对象 |
getDeclaredFields() | 获得所有属性对象 |
方法 | 用途 |
getAnnotation(Class annotationClass) | 返回该类中与参数类型匹配的公有注解对象 |
getAnnotations() | 返回该类所有的公有注解对象 |
getDeclaredAnnotation(Class annotationClass) | 返回该类中与参数类型匹配的所有注解对象 |
getDeclaredAnnotations() | 返回该类所有的注解对象 |
方法 | 用途 |
getConstructor(Class...<?> parameterTypes) | 获得该类中与参数类型匹配的公有构造方法 |
getConstructors() | 获得该类的所有公有构造方法 |
getDeclaredConstructor(Class...<?> parameterTypes) | 获得该类中与参数类型匹配的构造方法 |
getDeclaredConstructors() | 获得该类所有构造方法 |
方法 | 用途 |
getMethod(String name, Class...<?> parameterTypes) | 获得该类某个公有的方法 |
getMethods() | 获得该类所有公有的方法 |
getDeclaredMethod(String name, Class...<?> parameterTypes) | 获得该类某个方法 |
getDeclaredMethods() | 获得该类所有方法 |
方法 | 用途 |
isAnnotation() | 如果是注解类型则返回true |
isAnnotationPresent(Class<? extends Annotation> annotationClass) | 如果是指定类型注解类型则返回true |
isAnonymousClass() | 如果是匿名类则返回true |
isArray() | 如果是一个数组类则返回true |
isEnum() | 如果是枚举类则返回true |
isInstance(Object obj) | 如果obj是该类的实例则返回true |
isInterface() | 如果是接口类则返回true |
isLocalClass() | 如果是局部类则返回true |
isMemberClass() | 如果是内部类则返回true |
Field代表类的成员变量(成员变量也称为类的属性)。
方法 | 用途 | |
equals(Object obj) | 属性与obj相等则返回true | |
get(Object obj) | 获得obj中对应的属性值 | |
set(Object obj, Object value) 设置obj中对应属性值 |
Method代表类的方法。
方法 | 用途 | |
invoke(Object obj, Object... args) | 传递object对象及参数调用该对象对应的方法 |
Constructor代表类的构造方法。
方法 | 用途 |
newInstance(Object... initargs) | 根据传递的参数创建类的对象 |
示例:
Person p = new Person(); p.setAge("1"); p.setName("zs"); p.setSex("0"); p.show(); p.display("china");
//Class<T> 这里可以使用泛型 Class clazz = Person.class; Person person = (Person) clazz.newInstance(); person.setSex("1"); ; person.setName("zs2"); person.setAge("2"); person.show(); person.display("Keroar");
使用反射动态的设置
Person person2 = (Person) clazz.newInstance(); //这种这只属性只能是public修饰的 Field field = clazz.getField("name"); field.set(person2, " 我是名字 "); System.out.println(person2); //private 或者 其他的需要设置访问权限 Field field1 = clazz.getDeclaredField("sex"); field1.setAccessible(true); field1.set(person2, "我是sex"); System.out.println(person2);
3 通过反射获取指定类的指定方法属性
获取方法 根据方法名 以及 parameterTypes 可变参数类型,没有参数就不写了
public Method getMethod(String name, Class<?>... parameterTypes)
1 获取设置除public修饰的 //调用传参方法 display Method method = clazz.getMethod("show"); method.invoke(person2); //2 同样 设置除public 之外的访问 //public Method getDeclaredMethod(String name, Class<?>... parameterTypes) // method.setAccessible(true); //获取方法上注解以及方法参数类型 返回类型 方法名 System.out.println("方法名 : " + method.getName()); System.out.println("参数类型 : " + method.getTypeParameters()); System.out.println("返回类型 : " + method.getReturnType()); //System.out.println("返回所有注解 : " + method.getAnnotations()); //System.out.println("返回所有参数注解 : " + method.getParameterAnnotations()); System.out.println("返回方法参数注解 : " + method.getAnnotation(clazz)); //调用方法 //public Object invoke(Object obj, Object... args) obj 调用的对象,args 传入方法可变参数 //调用传参方法 display Method methods = clazz.getMethod("display", String.class); methods.invoke(person2, "我是传入的国籍参数");
每一个运行类只加载一次
有了Class 实例才能进行
1 创建对象 2 获取类中所有的属性 方法 父类 内部类 构造 异常 等等... 3 调用属性和方法等 4 反射的应用 : 动态代理
通过调用 getClass , 获取 Person 类运行时类的calss对象
Class pClass = p.getClass(); System.out.println(pClass);
//1 通过运行类对象调用 getClass 方法 //2 调用本身.class 属性 Class clazz1 = Person.class; Class clazz2 = String.class; // 3 通过Class 静态方法获取 Class<?> aClass = Class.forName("reflection.Person"); System.out.println(aClass); // 4 类的加载器 ClassLoader classLoader = this.getClass().getClassLoader(); Class<?> aClass1 = classLoader.loadClass("reflection.Person"); System.out.println(aClass1);
除了加载类之外类加载器可以用来干嘛?
//1 ClassLoader classLoader = aClass.getClassLoader(); // 加载类 // classLoader.loadClass(); //例如读取jdbc 配置文件 String name = "jdbc.properties"; InputStream resourceAsStream = classLoader.getResourceAsStream(name); Properties properties = new Properties(); properties.load(resourceAsStream); String property = properties.getProperty("key");
package reflection; public class Person { public String name; private String age; private String sex; // 0 男 1 女 public Person(String name, String age, String sex) { this.name = name; this.age = age; this.sex = sex; } public Person(String name) { this.name = name; } public Person(String name, String age) { this.name = name; this.age = age; } public Person(){ } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAge() { return age; } public void setAge(String age) { this.age = age; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } public void show(){ System.out.println(" emmmmmmm "); } public void display(String contry){ System.out.println(" 我的国籍是 " + contry); } @Override public String toString() { return "Person{" + "name='" + name + '/'' + ", age='" + age + '/'' + ", sex='" + sex + '/'' + '}'; } }