线上跑了个应用,总感觉运行结果和你预期的有出入,而且代码似乎和你本地的也不一致。这可咋办,一咬牙,一跺脚停机重新上线吗?
那是够折腾的,我们来了解几种不停机来查看 Java 应用Class 源码的方式。
像程咬金的功夫一样,针对Class 源码查看,我们也有三板斧。
一般我们要查看一个运行中程序的 Class 源码,大致会经过这样三步:
一、定位,找到这个Class
二、将远程 的 class 文件下载到本地
三、反编译,查看源码
这的过程,应该都很清楚,我们重点来看第一第二步,怎样找到这个class并下载到本地。
对于找Class 来说,如果是我们自己开发的代码,而不是编译时或者运行时增强以及生成的,那直接在远程服务器搜索到并下载到本地就OK了。
像那些编译时、以及运行时增强过的Class,甚至动态生成的Class,比如像OpenJPA、Spring/Cglib 的动态代理, 上面的办法就无能为力了。
这种情况,可以直接用JDK 的神器 SA 工具(Java虚拟机的显微镜 Serviceability Agent),在 Class browser 里直接搜索,然后选择 create .class file 文件就会默认保存到 SA Jar 所在的目录。不过缺点是,SA连接时会把应用挂起,直到断开。
还有一种方法是各种增强的类库,都默认提供了保存生成类的配置,如果此选项打开,就会把生成的、增强的class 写到本地。比如像Cglib,在System.property里设置即可。
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, ".");
Cglib 运行时判断配置,打开就会把生成类的 class 二进制文件写到本地。
如果类库没有或者你没找到这个选项,又确实想要查看,那可以 Attach 一个 Java Agent 到JVM上。用Instrument或者ByteBuddy免 Jar 包挂上去,参考之前的文章(不用Jar 包的Agent?几行代码实现运行时增强)
有了Agent 的 transformer,大显身手就看你的了。
本文转载自微信公众号「Tomcat那些事儿 」,可以通过以下二维码关注。转载本文请联系Tomcat那些事儿公众号。