反射
概念
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
在程序执行过程中动态的调用没有书写调用其他类的代码.
类加载器
将编译好的class二进制字节码文件交由jvm进行执行
三种默认类加载器
**启动类加载器:**将启动jvm的class文件进行加载执行
**扩展加载器:**将java运行基本jar包进行加载执行
**系统加载器:**将我们编写的class文件进行加载执行
类对象Class
java中万物皆对象,在使用反射时,将所有自定义类都当做类对象,都是class 的对象
获取类对象的三种方式
Class c1=Class.forName(“类名”);
使用Class类静态方法内部通过类加载器加载指定类class对象
Class c2=类名.class;
通过指定类class属性获取指定类class对象
Class c3=obj.getClass();
通过指定类对象的getClass方法获取指定class对象
// Class可以代表所有类与Object不同的是object是所有类的父类
// 所有类都是Class类的对象
// 获取一个类class对象的方式
// 1、使用Class类静态方法获取
Class c1 = Class.forName("com.yunhe.day1106.Student");
// 2、使用类的class属性获取(这种方式获取class对象必须声明变量保存)
Class c2 = Student.class;
// 3、使用指定类对象的getClass方法获取
Student s = new Student();
Class c3 = s.getClass();
Object newInstance = c1.newInstance();//调用指定class对象代表类的无参构造方法
System.out.println(newInstance);
反射获取构造方法
通过clas类提供的获取构造方法的方法,返回构造方法对象代表指定类的构造方法
Constructor getConstructor(Class<?>… parameterTypes)
返回一个 Constructor 对象,它反映此 Class 对象所表示的类的指定公共构造方法。
Constructor<?>[] getConstructors()
返回一个包含某些 Constructor 对象的数组,这些对象反映此 Class 对象所表示的类的所有公共构造方法。
Constructor getDeclaredConstructor(Class<?>… parameterTypes)
返回一个 Constructor 对象,该对象反映此 Class 对象所表示的类或接口的指定构造方法。
Constructor<?>[] getDeclaredConstructors()
返回 Constructor 对象的一个数组,这些对象反映此 Class 对象表示的类声明的所有构造方法。
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
//通过反射获取构造方法
public class ConstructorTest {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException,
InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
Class<?> c = Class.forName("com.yunhe.day1106.Student");
// 通过反射获取指定类的class对象
// 通过class对象获取指定类的构造方法对象
// getConstructor(Class<?>... parameterTypes)
// 通过参数获取指定参数个数与类型的构造方法
Constructor<?> c1 = c.getConstructor();// 无参公开的构造方法
Object c1o = c1.newInstance();
// System.out.println(c1o);
// 执行构造方法 并传入执行时需要的数据
Constructor<?> c2 = c.getConstructor(String.class);// 获取公开的参数类型为String的构造方法
Object c2o = c2.newInstance("张三");
// System.out.println(c2o);
// Constructor<?>[] getConstructors()
// 获取当前class对象代表类的所有公开的构造方法
Constructor<?>[] constructors = c.getConstructors();
// for (Constructor<?> constructor : constructors) {
// System.out.println(constructor);
// }
// Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
// 获取指定参数的构造方法(包括私有构造方法)
Constructor<?> c3 = c.getDeclaredConstructor(String.class, int.class);
c3.setAccessible(true);// 为私有构造方法赋权
Object c3o = c3.newInstance("张三", 18);
System.out.println(c3o);
// Constructor<?>[] getDeclaredConstructors()
// 返回 Constructor 对象的一个数组,这些对象反映此 Class 对象表示的类声明的所有构造方法。
Constructor<?>[] declaredConstructors = c.getDeclaredConstructors();
for (Constructor<?> constructor : declaredConstructors) {
System.out.println(constructor);
}
}
}
反射获取属性
Field getField(String name)
返回一个 Field 对象,它反映此 Class 对象所表示的类或接口的指定公共成员字段。
Field[] getFields()
返回一个包含某些 Field 对象的数组,这些对象反映此 Class 对象所表示的类或接口的所有可访问公共字段。
Field getDeclaredField(String name)
返回一个 Field 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明字段。
Field[] getDeclaredFields()
返回 Field 对象的一个数组,这些对象反映此 Class 对象所表示的类或接口所声明的所有字段。
import java.lang.reflect.Field;
import java.util.Arrays;
//通过反射获取属性
public class FieldTest {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException {
Class<?> c = Student.class;
Object o = c.newInstance();
// Field getField(String name)
// 获取指定class对象所代表类的指定名字的公开属性
Field field = c.getField("name");
//System.out.println(field);
// Field[] getFields()
// 返回一个包含某些 Field 对象的数组,这些对象反映此 Class 对象所表示的类或接口的所有可访问公共字段。
Field[] fields = c.getFields();
//System.out.println(Arrays.toString(fields));
// Field getDeclaredField(String name)
// 获取指定class对象代表的类的指定名字的属性对象(包含私有属性)
Field declaredField = c.getDeclaredField("age");
//System.out.println(declaredField);
// Field[] getDeclaredFields()
// 返回 Field 对象的一个数组,这些对象反映此 Class 对象所表示的类或接口所声明的所有字段。
Field[] declaredFields = c.getDeclaredFields();
//System.out.println(Arrays.toString(declaredFields));
System.out.println(o);
//Field属性对象的使用
//私有属性使用前进行赋权操作
declaredField.setAccessible(true);
//set方法为指定对象进行赋值
declaredField.set(o, 18);
field.set(o, "张三");
//get方法获取指定对象属性的属性
Object object = field.get(o);
System.out.println(object);
}
}
反射获取方法
Method getMethod(String name, Class<?>… parameterTypes)
返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法。
Method[] getMethods()
返回一个包含某些 Method 对象的数组,这些对象反映此 Class 对象所表示的类或接口(包括那些由该类或接口声明的以及从超类和超接口继承的那些的类或接口)的公共 member 方法。
Method getDeclaredMethod(String name, Class<?>… parameterTypes)
返回一个 Method 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明方法。
Method[] getDeclaredMethods()
返回 Method 对象的一个数组,这些对象反映此 Class 对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。
import java.lang.reflect.Method;
public class MethodTest {
public static void main(String[] args) throws Exception {
Class<?> c = Student.class;
Object o = c.newInstance();
// Method getMethod(String name, Class<?>... parameterTypes)
// 返回一个 Method 对象,获取指定名称指定参数列表的方法对象(公开的)
Method method = c.getMethod("setName", String.class);
// System.out.println(method);
// Method[] getMethods()
// 返回一个包含某些 Method 对象的数组,这些对象反映此 Class
// 返回所有公开的方法(包括继承父类的方法)
Method[] methods = c.getMethods();
// for (Method method2 : methods) {
// System.out.println(method2);
// }
// Method getDeclaredMethod(String name, Class<?>... parameterTypes)
// 获取指定方法(包括私有)
// 获取的是在当前类中定义的方法
Method declaredMethod = c.getDeclaredMethod("a",String.class);
// System.out.println(declaredMethod);
// Method[] getDeclaredMethods()
// 返回 Method 对象的一个数组,这些对象反映此 Class
// 对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。
Method[] declaredMethods = c.getDeclaredMethods();
// for (Method method2 : declaredMethods) {
// System.out.println(method2);
// }
//方法的执行
//私有方法执行前需要赋权
method.invoke(o, "张三");
declaredMethod.setAccessible(true);
//使用方法对象invole方法调用传入调用的对象与参数(无参则不写)
declaredMethod.invoke(o, "asd");
}
}
总结
反射的功能:通过预先一定格式的代码书写,可以在程序运行过程中动态的进行类的使用
反射的使用:
1、获取指定类的class对象
①使用类加载器进行加载指定类的全路径
②使用类的class属性进行获取
③使用类对象的getclass方法进行获取
2、构造方法
①构造方法的获取:四种方法进行构造方法的获取
②构造方法的使用:赋权、创建对象
3、属性
①属性的获取:四种方法进行属性的获取(注意获取属性时是否可以获取继承的属性)
②属性的使用:赋权、赋值、获取值(注意需要传入赋值的对象与值)
4、方法
①方法的获取:四种方式进行方法的获取(注意获取方法时是否可以获取继承的方法)
②方法的使用:赋权、执行(注意需要传入执行方法的对象与参数)