当前位置:首页 > 技术知识 > 正文内容

1000个字带你一次性搞懂JavaAgent技术,反正我是彻底服了

maynowei2周前 (08-19)技术知识5

JavaAgent技术

JavaAgent是一种特殊的Java程序,是Instrumentation的客户端。它与普通Java程序通过main方法启动不同,JavaAgent并不是一个可以单独启动的程序,它必须依附在一个Java应用程序(JVM)上,与主程序运行在同一个进程中,通过Instrumentation API与虚拟机交互。

JVM启动时静态加载

对于JVM启动时加载的Agent模块代码,Instrumentation会通过premain方法传入代理程序,premain方法会在调用程序main方法之前被调用,同时Instrumentation包含agentmain方法实现字节码改写,二者的区别如下:

● premain 方 法 用 于 在 启 动 时 , 在 类 加 载 前 定 义 类 的TransFormer(转化器),在类加载的时候更新对应的类的字节码。

● agentmain方法用于在运行时进行类的字节码的修改,步骤分为注册类的TransFormer调用和retransformClasses函数进行类的重加载。

premain方法与agentmain方法相比有很大的局限性。premain方法仅限于应用程序的启动时,即main函数执行前。此时还有很多类没有被加载,而这些类使用premain方法是无法实现字节码改写的。

目前,主流的基于探针的监控系统都是基于这种方式实现的对应用的无侵入监控。我们知道程序的入口是main方法,而premain方法代表了在程序正式启动之前执行的动作,它同时具备类似AOP的能力。

Transformer提供字节码文件流转化的能力,如下图所示是Class文件转换图。

字节码改写

如上图所示,任何Class文件在加载时,都要经过premain这一代码转换环节。通过一系列的TransFormer转换,Class字节码文件流最终转变为我们期望的代码实现,然后被加载到JVM中。修改Class字节码文件流的动作是在Transformer中进行的。我们可以使用Javaassist技术修改字节码文件流(下一节介绍)。下面就是我们实现的一个类 , 实 现 了 带 Instrumentation 参 数 的 premain 方 法 。 调 用addTransformer方法对启动时所有的类进行拦截,示例代码如下:

JVM启动后动态Instrument机制

关于JVM启动后动态加载Agent的方法,Instrumentation会通过agentmain方法传入程序。agentmain方法在main函数开始运行后才被调用,其最大优势是可以在程序运行期间进行字节码的替换。

Attach API[1]实现动态注入的原理如下。

你的应用程序通过虚拟机提供的attach(pid)方法,可以将代理程序连接(attach)到一个运行中的Java进程上,之后便可以通过loadAgent(AgentJarPath)将Agent的jar包注入对应的进程,然后对应的进程会调用agentmain方法,如下图所示。

工程结构和上面premain的一样,编写AgentMainTest代码示例如下:

JavaAgent运行前启动加载代理程序的方法如下。

JavaAgent有两个启动时机,一个是在程序启动时通过-javaAgent参数启动代理程序;另一个是在程序运行期间通过Java Tool API中的Attach API动态启动代理程序。我们通过-javaAgent来指定我们编写的Agent的jar路径(./{Location}/Agent.jar)。这样在启动时,Agent就可以做定制化的字节码改动了。对于Spring Boot类内置容器的服务,可以使用下面方式:

在Tomcat启动时,它会读取CATALINA_OPTS环境变量,并将它加入启动命令中。在环境变量中添加如下信息:

Java程序运行后加载代理的方法如下。

程序启动之后,我们通过某种特定的手段加载Java Agent。这个特定的手段就是虚拟机的Attach API。这个API其实是JVM进程之间的沟通桥梁,它的底层通过Socket进行通信。JVM A可以发送一些指令给JVM B,JVM B收到指令之后,可以执行对应的逻辑,比如在命令行中经常使用的jstack、jcmd、jps等命令。因为是进程间通信,所以使用Attach API的也是一个独立的Java进程。下面是一个简单的实现,代码示例如下:

本文给大家讲解的内容是服务监控治理, JavaAgent技术

  1. 下篇文章给大家讲解的内容是服务监控治理,Javaassist技术
  2. 觉得文章不错的朋友可以转发此文关注小编;
  3. 感谢大家的支持!

相关文章

单打独斗的产品设计师工作流程总结

来人人都是产品经理【起点学院】,BAT实战派产品总监手把手系统带你学产品、学运营。我从入行开始就在一个做自己产品的小公司工作,到现在已经三年了。刚开始工作的时候什么也不懂,老板说让出效果图,就开始直接...

有了这份900多页的Android面试指南,你离大厂Offer还远吗?

前言对于大部分程序员来说,一线互联网是的工作经历是毕生的追求,实际上大厂对于学历的要求远远没有我们想象的那么高,近几年来,互联网公司更注重技术,所以提升自身技术水平才是斩获offer的制胜关键。一线互...

Win10桌面/手机版最深层次开发功能挖掘

IT之家讯 Win10开发者预览版为我们提供了一个Win10大框架的早期概览,使开发者与热心用户都可以提前感受Win10带来的新特性,尝试新工具,而作为开发者,最关心的莫过于Windows多平台通用应...

btrace 3.0 重磅新增 iOS 支持!免插桩原理大揭秘!

重磅更新btrace 是由字节跳动抖音基础技术团队自主研发的面向移动端的性能数据采集工具,它能够高效的助力移动端应用采集性能 Trace 数据,深入剖析代码的运行状况,进而辅助优化提升移动端应用的性能...

掌握C语言多线程:高效并发编程指南

一、多线程基础概念介绍多线程编程是现代软件开发中提高程序性能和响应性的重要技术。在C语言中,pthread(POSIX Threads)库是实现多线程编程的标准工具。本节将通俗易懂地介绍多线程的核心概...

Go语言进阶:时间轮(golang时间轮)

时间轮概念时间轮(Timing Wheel)是一种高效的定时任务调度数据结构,特别适合处理大量定时任务。它通过一个循环数组(轮盘)和多个槽位(buckets)来组织定时任务,每个槽位代表一个时间间隔。...