类加载过程的核心:任何一个类被系统使用时候都会将class文件加载进内存并创建一个Class对象,同时会初始化静态成员
类加载时机:
- 创建类的实例
- 访问类的静态变量,或者为静态变量赋值
- 调用类的静态方法
- 使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
- 初始化某个类的子类
- 直接使用java.exe命令来运行某个主类
负责人:类加载器,其分类有
BootStrap ClassLoader 根类加载器,负责Java核心类的加载,比如System,String等等,位于jre--lib--rt文件中
Extension ClassLoader 扩展加载器,负责JRE拓展包中的类加载,jre--lib--ext
System ClassLoader 系统加载器,负责在JVM启动时加载由java命令启动的class文件,以及classPath所指定的jar包和类
反射定义:
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象.而Class对象是在运行期间才能确定的,所以反射运用在运行期间。
对于每一种类,Class对象只有唯一的一个,但对象实例可以new多个。
获取反射对象有三种方式
Student student = new Student("Leo");
//one
Class clazz = student.getClass();
//two
Class clazz1 = Student.class;
//three
try {
Class clazz2 = Class.forName("Model_4.Student");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
开发中使用更多的是方式三,因为变化的是字符串,代码设计更灵活。
获取到Class对象后我们就操作该对象中的方法
public Constructor>[] getConstructors()//所有公共构造方法
public Constructor>[] getDeclaredConstructors()//所有构造方法
public Constructor getConstructor(Class... var1)//获取特定构造方法
public Constructor getDeclaredConstructor(Class... var1)
//from Constructor
public T newInstance(Object... var1)//创建该构造方法对应类型的实例,相当于new一个对象
public void setAccessible(boolean var1) 将能取消语言访问检查机制
eg:
Class clazz2 = null;
try {
clazz2 = Class.forName("Model_4.Student");
Constructor constructor = clazz2.getConstructor(String.class,int.class);
Object object = constructor.newInstance("Leo",24);
} catch (Exception e) {
e.printStackTrace();
}
//获取成员变量
public Field getField(String var1)
public Field getDeclaredField(String var1)
public Field[] getFields()
public Field[] getDeclaredFields()
eg:
try {
clazz2 = Class.forName("Model_4.Student");
Constructor constructor = clazz2.getConstructor(String.class);
Student object = (Student) constructor.newInstance("Leo");
System.out.println(object.getOfficialName());//print Leo
Field field = clazz2.getDeclaredField("officialName");
field.setAccessible(true);
field.set(object,"PDDDDD");
System.out.println(object.getOfficialName());//print PDDDDD
} catch (Exception e) {
e.printStackTrace();
}
//获取方法
public Method getMethod(String var1, Class... var2)
public Method getDeclaredMethod(String var1, Class... var2)
public Method[] getMethods()
public Method[] getDeclaredMethods()
try {
clazz2 = Class.forName("Model_4.Student");
Constructor constructor = clazz2.getConstructor(String.class);
Student object = (Student) constructor.newInstance("Leo");
Method method1 = clazz2.getMethod("setOfficialName",String.class);
method1.invoke(object,"PDDDDDD");
Method method2 = clazz2.getMethod("getOfficialName");
String string = (String) method2.invoke(object);
System.out.println(string);
} catch (Exception e) {
e.printStackTrace();
}
常规class加载模式是预先加载需要使用的Class对象(文章开头提到了加载实际),然后我们就可以通过new对象的方式使用对象,而反射是运行过程中才加载Class对象,在此时根据需求创建实例。
其他注意事项
对于方法:
getDeclaredMethods:所有本类中声明的方法,包括继承或实现过来的方法(只返回本类的实现而非父类的)
getMethods:所有声明的公共方法,包括继承或实现过来的公共方法(只返回本类的实现而非父类的),也包括所有父类的公共方法
getDeclaredMethod getMethod同理
invoke只能调用公共方法,如果需要调用default、protected、private方法,需要提前调用method.setAccessible(true);
子类覆盖父类方法时候,方法限定符发生改变,上述规则不失效,即获取方法时候只返回一个最靠近继承链新生末端的方法。
对于域:
getDeclaredFields:所有本类中声明的域
getFields:所有本类和父类和接口中的公共域,包括重名且重类型的(域不存在继承链覆盖)
getDeclaredField getField同理
默认只能操作公共域,如果需要获取或操作default、protected、private域,需要提前调用field.setAccessible(true);
调用getDeclaredField getField如果和父类或接口有重名的域,只返回一个最靠近继承链新生末端的域
反射会降低程序运行效率,主要是寻找方法和成员变量的过程比较缓慢,真正运行方法和操作成员变量的过程是不慢的。
- 在项目中使用反射
//设置配置文件,在项目中加载配置文件能够灵活实现功能,不需要时刻修改代码并重新编译项目
// 加载键值对数据
Properties prop = new Properties();
FileReader fr = new FileReader("class.txt");
prop.load(fr);
fr.close();
// 获取数据
String className = prop.getProperty("className");
String methodName = prop.getProperty("methodName");
// 反射
Class c = Class.forName(className);
Constructor con = c.getConstructor();
Object obj = con.newInstance();
// 调用方法
Method m = c.getMethod(methodName);
m.invoke(obj);
- 用反射跳出类型检查
//先看下面的代码
ArrayList list = new ArrayList<>();
list.add(10);
//通过编译后,JVM实际采用的class文件通过反编译,得到的是
ArrayList list = new ArrayList();
list.add(Integer.valueOf(10));
本质上,编译器把我们的Java代码转化成了上述形式进行使用,相当于我们本来就是这么写的。
通过查看ArrayList源码,我们发现,ArrayList构建时声明的泛型类型其实是Object,我们在编码时通过泛型声明,编译器会将泛型对象转化为特定的对象,本例子中即为Integer.valueOf(),同样为一个Object。可见泛型类型确认是发生在编译过程中。
但是,ArrayList源码反应,其Class对象始终没有确认泛型对象,保持Object类型,所以我们可以做如下操作:
ArrayList list = new ArrayList<>();
list.add(10);
try {
Class clz = list.getClass();
Method method = clz.getMethod("add",Object.class);
method.invoke(list,"pddddd");
method.invoke(list,"yjj");
method.invoke(list,new Student("leon"));
System.out.println(list);//调用的是ArrayList的toString方法,里面是迭代调用元素的toString方法
} catch (Exception e) {
e.printStackTrace();
}
- 暴力操作对象的私有数据
当项目中的文件没有提供对外操作的接口,,,
public class Tool {
public void setProperty(Object obj, String propertyName, Object value)
throws NoSuchFieldException, SecurityException,
IllegalArgumentException, IllegalAccessException {
// 根据对象获取字节码文件对象
Class c = obj.getClass();
// 获取该对象的propertyName成员变量
Field field = c.getDeclaredField(propertyName);
// 取消访问检查
field.setAccessible(true);
// 给对象的成员变量赋值为指定的值
field.set(obj, value);
}
}
动态代理
代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象
动态代理:在程序运行过程中产生的这个对象
而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以,动态代理其实就是通过反射来生成一个代理
在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理。我们有更强大的代理cglib
Proxy类中的方法创建动态代理类对象
public static Object newProxyInstance(ClassLoader loader,Class>[] interfaces,InvocationHandler h)
最终会调用InvocationHandler的方法
InvocationHandler
Object invoke(Object proxy,Method method,Object[] args)
Proxy类中创建动态代理对象的方法的三个参数;
ClassLoader对象,定义了由哪个ClassLoader对象来对生成的代理对象进行加载
Interface对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了
InvocationHandler对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个InvocationHandler对象上
public class KaSha implements UserDao {
ArrayList skills = new ArrayList<>();
int shuijin = 100;
@Override
public boolean addSkill(String skill) {
return skills.add(skill);
}
@Override
public void deleteAccessory(int number) {
shuijin = shuijin - number;
}
@Override
public void login() {
System.out.println(this.getClass().getName()+" has log in");
}
@Override
public void exit() {
System.out.println(this.getClass().getName()+" has log out");
}
}
public class HeroHandler implements InvocationHandler {
private UserDao mUserDao;
public HeroHandler(UserDao userDao) {
mUserDao = userDao;
}
@Override
public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
System.out.println("权限校验");
return method.invoke(mUserDao,objects);
}
}
public static void main(String[] args) {
UserDao userDao = new KaSha();
HeroHandler heroHandler = new HeroHandler(userDao);
UserDao userDao1 = (UserDao) Proxy.newProxyInstance(userDao.getClass().getClassLoader(),userDao.getClass().getInterfaces(),heroHandler);
userDao1.login();
userDao1.addSkill("皇族旗帜");
userDao1.deleteAccessory(2);
userDao1.exit();
}
//输出结果中每个方法的实现都会转化为HeroHandler的invoke方法实现。这样对于需要批量增加的操作,就不需要在重写子类实现,比如上述的“权限检查”。