名词解释:所谓热部署,就是在应用正在运行的时候升级软件,却不须要从新启动应用。java
对于Java应用程序来讲,热部署就是在运行时更新Java类文件。在基于Java的应用服务器实现热部署的过程当中,类装入器扮演着重要的角色
。大多数基于Java的应用服务器,包括EJB服务器和Servlet容器,都支持热部署。类装入器不能从新装入一个已经装入的类,但只要使用一个新的类装入器实例,就能够将类再次装入一个正在运行的应用程序
。web
咱们知道,如今大多数的web服务器都支持热部署,而对于热部署的实现机制,网上讲的却不够完善,下面咱们就Tomcat的热部署实现机制,讲解一下它是如何实现的:spring
Tomcat的容器实现热部署使用了两种机制:apache
Classloader重写,经过自定义classloader加载相应的jsp编译后的class到JVM中。tomcat
经过动态修改内存中的字节码,将修改过的class再次装载到JVM中。服务器
Tomcat经过org.apache.jasper.servlet.JasperLoader
实现了对jsp的加载,下面作个测试:
1. 新建一个web工程,并编写一个jsp页面,在jsp页面中输出该页面的classloader,<%System.out.print(this.getClass().getClassLoader());%>.
2. 启动web服务器,打开jsp页面,咱们能够看到后台输出,该jsp的classloader是JasperLoader的一个实例。
3. 修改jsp,保存并刷新jsp页面,再次查看后台输出,此classloader实例已经不是刚才那个了,也就是说tomcat经过一个新的classloader再次装载了该jsp。
4. 其实,对于每一个jsp页面tomcat都使用了一个独立的classloader来装载,每次修改完jsp后,tomcat都将使用一个新的classloader来装载它。app
关于如何使用自定义classloader来装载一个class这里就不说了,相信网上都能找到,JSP属于一次性消费,每次调用容器将建立一个新的实例,属于用完就扔的那种,可是对于这种实现方式却很难用于其它状况下,如如今咱们工程中不少都使用了单例,尤为是spring工程,在这种状况下使用新的classloader来加载修改后的类是不现实的,单例类将在内存中产生多个实例,并且这种方式没法改变当前内存中已有实例的行为,固然,tomcat也没经过该方式实现class文件的从新加载。jsp
Tomcat中的class文件是经过org.apache.catalina.loader. WebappClassLoader
装载的,一样咱们能够作个测试,测试过程与jsp测试相似,测试步骤就不说了,只说一下结果:ide
在热部署的状况下,对于被该classloader 加载的class文件,它的classloader始终是同一个WebappClassLoader
,除非容器重启了,相信作完这个实验你就不会再认为tomcat是使用一个新的classloader来加载修改过的class了,并且对于有状态的实例,以前该实例拥有的属性和状态都将保存,并在下次执行时拥有了新的class的逻辑,这就是热部署的神秘之处(其实每一个实例只是保存了该实例的状态属性,咱们经过序列化对象就能看到对象中包含的状态,最终的逻辑仍是存在于class文件中)。测试
下面的class重定义是经过:java.lang.instrument实现的
,具体可参考相关文档。
下面咱们看一下如何经过代理修改内存中的class字节码:
如下是一个简单的热部署代理实现类(代码比较粗糙,也没什么判断):
import java.lang.instrument.ClassFileTransformer; import java.lang.instrument.ClassFileTransformer; import java.lang.instrument.Instrumentation; import java.util.Set; import java.util.Timer; import java.util.TreeSet; public class HotAgent { protected static Set<String> clsnames=new TreeSet<String>(); public static void premain(String agentArgs, Instrumentation inst) throws Exception { ClassFileTransformer transformer =new ClassTransform(inst); inst.addTransformer(transformer); System.out.println("是否支持类的重定义:"+inst.isRedefineClassesSupported()); Timer timer=new Timer(); timer.schedule(new ReloadTask(inst),2000,2000); } }
import java.lang.instrument.ClassFileTransformer; import java.lang.instrument.ClassFileTransformer; importjava.lang.instrument.IllegalClassFormatException; import java.lang.instrument.Instrumentation; import java.security.ProtectionDomain; public class ClassTransform. implements ClassFileTransformer { private Instrumentation inst; protected ClassTransform(Instrumentation inst){ this.inst=inst; } /** * 此方法在redefineClasses时或者初次加载时会调用,也就是说在class被再次加载时会被调用, * 而且咱们经过此方法能够动态修改class字节码,实现相似代理之类的功能,具体方法可以使用ASM或者javasist, * 若是对字节码很熟悉的话能够直接修改字节码。 */ public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer)throws IllegalClassFormatException { byte[] transformed = null; HotAgent.clsnames.add(className); return null; } }
import java.lang.instrument.ClassDefinition; import java.io.InputStream; import java.lang.instrument.ClassDefinition; import java.lang.instrument.Instrumentation; import java.util.TimerTask; public class ReloadTask extends TimerTask { private Instrumentation inst; protected ReloadTask(Instrumentation inst){ this.inst=inst; } @Override public void run() { try{ ClassDefinition[] cd=new ClassDefinition[1]; Class[] classes=inst.getAllLoadedClasses(); for(Class cls:classes){ if(cls.getClassLoader()==null||!cls.getClassLoader().getClass().getName().equals("sun.misc.Launcher$AppClassLoader")) continue; String name=cls.getName().replaceAll("\\.","/"); cd[0]=new ClassDefinition(cls,loadClassBytes(cls,name+".class")); inst.redefineClasses(cd); } }catch(Exception ex){ ex.printStackTrace(); } } private byte[] loadClassBytes(Class cls,String clsname) throws Exception{ System.out.println(clsname+":"+cls); InputStream is=cls.getClassLoader().getSystemClassLoader().getResourceAsStream(clsname); if(is==null)return null; byte[] bt=new byte[is.available()]; is.read(bt); is.close(); return bt; } }
以上是基本实现代码,须要组件为:
1.HotAgent(预加载)
2.ClassTransform(在加载class的时候能够修改class的字节码),本例中没用到
3.ReloadTask(class定时加载器,以上代码仅供参考)
4.META-INF/MANIFEST.MF内容为:(参数一:支持class重定义;参数二:预加载类)
Can-Redefine-Classes: true
Premain-Class: agent.HotAgent
5.将以上组件打包成jar文件(到此,组件已经完成,下面为编写测试类文件)。
6.新建一个java工程,编写一个java逻辑类,并编写一个Test类,在该测试类中调用逻辑类的方法,下面看下测试类代码:
package test.redefine; public class Bean1 { public void test1(){ System.out.println("============================"); } }
package test.redefine; public class Test { public static void main(String[] args)throws InterruptedException { Bean1 c1=new Bean1(); while(true){ c1.test1(); Thread.sleep(5000); } } }
运行测试类:
java –javaagent:agent.jar test.redefine.Test
在测试类中,咱们使用了一个死循环,定时调用逻辑类的方法。咱们能够修改Bean1中的方法实现,将在不一样时间看到不一样的输出结果,关于技术细节也没什么好讲的了,相信你们都能明白。