SpringBoot热加载实现与类加载浅谈
热部署与热加载:
热部署和热加载都是基于类加载器实现的,热加载是服务器监听class等文件的改变然后对改变的文件进行局部加载,所以不会删除session,也不会释放内存。热部署就是全局部署,会清空session以及释放内存。
类加载机制:
java类加载器主要有:
根类加载器(Bootstrap ClassLoader) 一般加载%JAVA_HOME%/jre/lib下的一些jar包的class文件
扩展类加载器(Extension ClassLoader) 一般加载%JAVA_HOME%/jre/lib/ext中jar包的class文件
系统类加载器(System ClassLoader) 一般加载来自java命令的-classpath指定的jar包和类路径
自定义加载器(User ClassLoader) 用户自定义的加载器,默认的父加载器是系统类加载器
这些加载器有一个双亲委派的机制,就是说,当一个加载器接收到请求,他会委托他的父类加载器去加载。如果父类找不到,他才会去加载。举个栗子,java.lang.String 类,如果是系统类加载器接到请求,他会委托给扩展类加载器,扩展类加载器会委托给根类加载器,因为java.lang.String.class是在%JAVA_HOME%/jre/lib/rt.jar下,所以最终由根类加载器加载到JVM内存
类加载的阶段:
加载:将类的class文件读入内存,并创建一个java.lang.Class对象放入方法区
验证:检测被加载的类是否有正确的内部结构,会不会威胁虚拟机的安全等
准备:为static修饰的变量分配内存,并设置初始值,int型的是0,如果是static final修饰的,则进入常量池
解析:将类的二进制数据中的符号引用替换为直接引用。
初始化:当初始化一个类的时候,如果发现其父类还没有进行过初始化、则需要先触发其父类的初始化,将准备阶段的初始值完成赋值。初始化运行主类,比如main方法。
tomcat热部署实现方式:
1.直接把项目的web文件夹放在webapps里
2.在tomcat/conf/server.xml的<host>里添加<context>
<Context docBase="xxx" path="/xxx" reloadable="true"/>
其中docBase是你项目的web文件夹所在的目录,path是用来在浏览器访问项目的虚拟路径。
3.在tomcat/conf/Catalina/localhost/下添加一个xml,内容如下
<?xml version="1.0" encoding="UTF-8"?>
<Context docBase="xxx" reloadable="true"/>
其中docBase和reloadable跟第二步一样,而项目访问的虚拟路径是这个xml文件的文件名。
SpringBoot实现热加载
1.springloaded方式: 通过在maven的plug里加springloaded依赖
注意这种方式只能通过maven的方式去启动。在idea编辑器里可以配置maven启动
注意maven启动命令是mvn spring-boot:run 但是idea已经默认前面加了mvn,所以这里配置不需要再写mvn了。
2.通过spring-boot-devtools方式: 在maven添加依赖
注意用idea编辑器的,idea实现热加载还需要修改两个地方配置
这个要勾选上
然后按shift+alt+ctrl+/
这个勾选上。到此就ok了。
自定义加载器实现热加载
用户自定义加载器需要继承ClassLoader,实现原理就是通过一个线程去监听文件的修改时间,然后重写findClass方法,把文件以流的形式读进来,然后调defineClass方法。在JDK1.2之后,双亲委派模式已经被引入到类加载体系中,因此不建议重写loadClass方法,只需要重写findClass就可以了
这里是重写findClass核心代码。
原作者:catalina_ 转载来源:https://www.jianshu.com/p/cb6421d69383 非常感谢原作者的杰出贡献,转载文章仅供学习参考,如有侵权,敬请联系,以便及时删除