字节码编程之bytebuddy结合javaagent支持多种监控方式

写在前面

打印方法执行耗时是监控,获取程序运行的JVM信息是监控,链路追踪也是监控。

本文看下如何实现一个通用的监控解决方案。

1:程序

定义premain:

package com.dahuyou.multi.monitor;import com.dahuyou.multi.monitor.factory.PluginFactory;
import com.dahuyou.multi.monitor.linktrace.LinkTraceAdvice;
import com.dahuyou.multi.monitor.plugin.IPlugin;
import com.dahuyou.multi.monitor.plugin.InterceptPoint;
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.asm.Advice;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.DynamicType;
import net.bytebuddy.matcher.ElementMatchers;
import net.bytebuddy.utility.JavaModule;import java.lang.instrument.Instrumentation;
import java.util.List;public class MyPreMain {public static void premain(String agentArgs, Instrumentation inst) {System.out.println("Java agent配置参数:" + agentArgs);System.out.println("============================");System.out.println("大忽悠🐂B监控系统开始工作了!");System.out.println("============================");AgentBuilder agentBuilder = new AgentBuilder.Default();// 获取所有可用的监控插件(jvm,链路追踪,执行耗时等)List<IPlugin> pluginGroup = PluginFactory.pluginGroup;for (IPlugin plugin : pluginGroup) {InterceptPoint[] interceptPoints = plugin.buildInterceptPoint();for (InterceptPoint point : interceptPoints) {AgentBuilder.Transformer transformer= (builder, typeDescription, classLoader, javaModule) -> {builder = builder.visit(Advice.to(plugin.adviceClass()).on(point.buildMethodsMatcher()));return builder;};agentBuilder = agentBuilder.type(point.buildTypesMatcher()).transform(transformer).asDecorator();}}// 将bytebuddy的插桩逻辑安装到instrumentagentBuilder.with(new AgentBuilder.Listener() {@Overridepublic void onDiscovery(String s, ClassLoader classLoader, JavaModule javaModule, boolean b) {}@Overridepublic void onTransformation(TypeDescription typeDescription, ClassLoader classLoader, JavaModule javaModule, boolean b, DynamicType dynamicType) {
//                System.out.println("onTransformation:" + typeDescription);}@Overridepublic void onIgnored(TypeDescription typeDescription, ClassLoader classLoader, JavaModule javaModule, boolean b) {}@Overridepublic void onError(String s, ClassLoader classLoader, JavaModule javaModule, boolean b, Throwable throwable) {}@Overridepublic void onComplete(String s, ClassLoader classLoader, JavaModule javaModule, boolean b) {}}).installOn(inst);}
}

这里将不同的监控类型抽象为插件,定义了监控插件接口类:

package com.dahuyou.multi.monitor.plugin;/*** 监控插件接口*/
public interface IPlugin {/*插件名称*/String pluginName();/*插件的监控点(即监控哪些类的哪些方法)*/InterceptPoint[] buildInterceptPoint();/*执行具体监控的切面类*/Class adviceClass();}

本文实现,打印JVM,和链路追踪两种监控方式,实现类如下:

package com.dahuyou.multi.monitor.plugin.concrete;import com.dahuyou.multi.monitor.jvmusage.JVMUsageAdvice;
import com.dahuyou.multi.monitor.plugin.IPlugin;
import com.dahuyou.multi.monitor.plugin.InterceptPoint;
import net.bytebuddy.description.method.MethodDescription;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.matcher.ElementMatcher;
import net.bytebuddy.matcher.ElementMatchers;/*** jvm使用情况插件*/
public class JvmPlugin implements IPlugin {@Overridepublic String pluginName() {return "jvmusage";}@Overridepublic InterceptPoint[] buildInterceptPoint() {return new InterceptPoint[]{new InterceptPoint() {@Overridepublic ElementMatcher<TypeDescription> buildTypesMatcher() {return ElementMatchers.nameStartsWith("com.dahuyou.multi.monitor");}@Overridepublic ElementMatcher<MethodDescription> buildMethodsMatcher() {return ElementMatchers.named("method1").or(ElementMatchers.named("method2")).or(ElementMatchers.named("method3"));}}};}@Overridepublic Class adviceClass() {return JVMUsageAdvice.class;}}
package com.dahuyou.multi.monitor.plugin.concrete;import com.dahuyou.multi.monitor.jvmusage.JVMUsageAdvice;
import com.dahuyou.multi.monitor.linktrace.LinkTraceAdvice;
import com.dahuyou.multi.monitor.plugin.IPlugin;
import com.dahuyou.multi.monitor.plugin.InterceptPoint;
import net.bytebuddy.description.method.MethodDescription;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.matcher.ElementMatcher;
import net.bytebuddy.matcher.ElementMatchers;/*** 链路追踪插件*/
public class LinkTracePlugin implements IPlugin {@Overridepublic String pluginName() {return "linktrace";}@Overridepublic InterceptPoint[] buildInterceptPoint() {return new InterceptPoint[]{new InterceptPoint() {@Overridepublic ElementMatcher<TypeDescription> buildTypesMatcher() {return ElementMatchers.nameStartsWith("com.dahuyou.multi.monitor");}@Overridepublic ElementMatcher<MethodDescription> buildMethodsMatcher() {return ElementMatchers.named("method1").or(ElementMatchers.named("method2")).or(ElementMatchers.named("method3"));}}};}@Overridepublic Class adviceClass() {return LinkTraceAdvice.class;}}

通过插件工厂PluginFactory来维护所有可用的监控插件:

package com.dahuyou.multi.monitor.factory;import com.dahuyou.multi.monitor.plugin.IPlugin;
import com.dahuyou.multi.monitor.plugin.concrete.JvmPlugin;
import com.dahuyou.multi.monitor.plugin.concrete.LinkTracePlugin;
import java.util.ArrayList;
import java.util.List;/*** 可用监控插件工厂类*/
public class PluginFactory {public static List<IPlugin> pluginGroup = new ArrayList<>();static {//链路追踪监控pluginGroup.add(new LinkTracePlugin());//Jvm监控插件pluginGroup.add(new JvmPlugin());}}

接口InterceptPoint,用来抽象监控点,接口定义如下:

package com.dahuyou.multi.monitor.plugin;import net.bytebuddy.description.method.MethodDescription;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.matcher.ElementMatcher;/*** 插件监控点接口,设置插件要监控的类和要监控的方法*/
public interface InterceptPoint {/*设置插件要监控的类*/ElementMatcher<TypeDescription> buildTypesMatcher();/*设置插件要监控的方法*/ElementMatcher<MethodDescription> buildMethodsMatcher();}

核心逻辑就是这些了,详细的大家感兴趣的话还是看源码。

接着来打包:
在这里插入图片描述
测试类:

package com.dahuyou.multi.monitor.test;public class ApiTest {public static void main(String[] args) {//线程一new Thread(() -> new ApiTest().method1(), "线程思密达").start();
//        new Thread(() -> new ApiTest().method1(), "线程萨瓦迪卡").start();//        new ApiTest().method1();}public void method1() {System.out.println("测试结果:hi1");method2();}public void method2() {System.out.println("测试结果:hi2");method3();}public void method3() {System.out.println("测试结果:hi3");}}

配置javaagent:
在这里插入图片描述
运行测试:
在这里插入图片描述

写在后面

参考文章列表

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/48293.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据库内核研发学习之路(五)创建postgres系统表

写在前面 在使用postgres的时候&#xff0c;有很多表是我们一开始安装好数据库就存在的&#xff0c;这些表称为系统表&#xff0c;他们记载一些数据库信息&#xff0c;比如我们做运维工作常用的pg_stat_activity;我们在数据库中查询这张表可以发现他存储了一些数据库连接信息。…

GO:Socket编程

目录 一、TCP/IP协议族和四层模型概述 1.1 互联网协议族&#xff08;TCP/IP&#xff09; 1.2 TCP/IP四层模型 1. 网络访问层&#xff08;Network Access Layer&#xff09; 2. 网络层&#xff08;Internet Layer&#xff09; 3. 传输层&#xff08;Transport Layer&#…

WPF+Mvvm 项目入门完整教程(一)

WPF+Mvvm 入门完整教程一 创建项目MvvmLight框架安装完善整个项目的目录结构创建自定义的字体资源下载更新和使用字体资源创建项目 打开VS2022,点击创建新项目,选择**WPF应用(.NET Framework)** 创建一个名称为 CommonProject_DeskTop 的项目,如下图所示:MvvmLight框架安装…

【JavaScript 算法】双指针法:高效处理数组问题

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 一、算法原理二、算法实现示例问题1&#xff1a;两数之和 II - 输入有序数组示例问题2&#xff1a;反转字符串中的元音字母注释说明&#xff1a; 三、应用场景四、总结 双指针法&#xff08;Two Pointer Technique&#xff…

深入理解Java并发线程阻塞唤醒类LockSupport

LockSupprot 用来阻塞和唤醒线程&#xff0c;底层实现依赖于Unsafe类 该类包含一组用于阻塞和唤醒线程的静态方法&#xff0c;这些方法主要是围绕 park 和 unpark 展开 public class LockSupportDemo1 {public static void main(String[] args) {Thread mainThread Thread.cu…

微信小程序(百战商城)的实战项目的首页的制作及讲解

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…

【人工智能】基于香橙派AIpro和昇腾AI计算芯片的面部口罩检测(详细教程)

目录 前言 1.介绍开发板 2.应用场景 3.安装操作系统 3.1 下载工具 3.2 烧录系统 3.3 系统装载 4.配置操作系统 4.1 登录系统账户 4.2 配置网络连接 4.3 查看设备网络 4.4 配置远程连接 5.部署目标检测应用 5.1 准备运行环境 5.2 模型二次训练 ​5.3 热成像温度…

数据结构 day4

目录 思维导图&#xff1a; 学习内容&#xff1a; 1. 链表的引入 1.1 顺序表的优缺点 1.1.1 优点 1.1.2 不足 1.1.3 缺点 1.2 链表的概念 1.2.1 链式存储的线性表叫做链表 1.2.2 链表的基础概念 1.3 链表的分类 2. 单向链表 2.1 节点结构体类型 2.2 创建链表 2.…

pikachu之暴力破解

1基于表单的暴力破解 随便输入然后抓包 选中添加账号密码 添加分别添加payload1&#xff0c;2&#xff0c;的字典 开始攻击 2验证码绕过on server 和基于表单的暴力破解相比&#xff0c;多了一个验证码功能 这个验证码是前端的验证码&#xff08;和前面那个一样选中添加账号密码…

Java小技能:多级组织机构排序并返回树结构(包含每个层级的子节点和业务数据集合)

文章目录 引言I 实体定义1.1 部门1.2 用户组织机构中间表1.3 树状DTOII 抽取组织机构排序方法2.1 树状排序方法2.2 案例III 查询条件构建3.1 根据部门进行权限控制3.2 注入风险引言 需求: 根据组织机构进行数据授权控制,例如控制船舶、船舶设备、摄像头、港区查看权限。 一…

kettle从入门到精通 第七十六课 ETL之kettle kettle连接hive教程

1、群里有小伙伴询问kettle连接hive的demo&#xff0c;今天抽点时间整理下。其实kettle连接hive和连接mysql数据库也是一样的。 1&#xff09;kettle中的lib目录下放hive驱动jar&#xff0c;这里我使用的是kyuubi-hive-jdbc-shaded-1.9.0.jar。 2&#xff09;设置hive连接参数…

pytorch学习(九)激活函数

1.pytorch常用激活函数如下&#xff1a; #ReLU激活函数 #Leaky ReLU激活函数 #Sigmoid激活函数 #Tanh激活函数 #Softmax激活函数 #Softplus2.代码 import torch.nn as nn import torch import numpy from torch.utils.tensorboard import SummaryWriterwriter SummaryWriter…

ModuleNotFoundError: No module named ‘lime‘,lime。 安装 LIME库

LIME LIME 的作用安装 LIME示例代码详细解释 总结 LIME&#xff08;Local Interpretable Model-agnostic Explanations&#xff0c;局部可解释不可知模型&#xff09;是一个Python库&#xff0c;用于解释机器学习模型的预测结果。它通过构建一个简单的、本地的可解释模型来近似…

【BUG】已解决:ModuleNotFoundError: No module named ‘torch‘

已解决&#xff1a;ModuleNotFoundError: No module named ‘torch‘ 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 欢迎来到我的主页&#xff0c;我是博主英杰&#xff0c;211科班出身&#xff0c;就职于医疗科技公司&#xff0c;热衷分享知识&#xff0c;武汉城市…

Unity UGUI 之EventSystem

本文仅作学习笔记与交流&#xff0c;不作任何商业用途 本文包括但不限于unity官方手册&#xff0c;唐老狮&#xff0c;麦扣教程知识&#xff0c;引用会标记&#xff0c;如有不足还请斧正 1.EventSystem是什么&#xff1f; 有需要请查看手册&#xff1a;Unity - 手册&#xff1…

2024.7.19最新详细的VMware17.0.0安装

VM官网VMware - Delivering a Digital Foundation For Businesses。现在官网无法下载&#xff0c;点击会跳转到https://access.broadcom.com/default/ui/v1/signin/ 要注册一个账号&#xff1a; 注册登录以后&#xff0c;点击Please select your identity provider. - Support …

昇思25天学习打卡营第2天 | 快速入门

在快速发展的人工智能领域&#xff0c;深度学习已经成为数据分析和模式识别的核心技术。作为一名深度学习初学者&#xff0c;我有幸通过MindSpore平台进行了实战演练&#xff0c;从数据预处理到模型训练与测试&#xff0c;再到模型保存与加载&#xff0c;经历了一次完整的深度学…

基于SpringBoot+Vue的校园台球厅设备管理系统(带1w+文档)

基于SpringBootVue的校园台球厅设备管理系统(带1w文档) 基于SpringBootVue的校园台球厅设备管理系统(带1w文档) 本次设计任务是要设计一个校园台球厅人员与设备管理系统&#xff0c;这个系统能够满足校园台球厅人员与设备的管理及用户的校园台球厅人员与设备管理功能。系统的主…

彻底卸载360安全卫士的方法

法一&#xff1a; 按下WindowsR键&#xff0c;并输入msconfig, 在“引导”选项卡中选择“安全引导”&#xff0c;并重新启动进入安全模式。此时&#xff0c;重复第一种方法“应用和功能”-“360安全卫士”-“卸载”&#xff0c;在弹出的对话框中残忍的拒绝它的各种令人发指的无…

go-微服务的设计概括

一、微服务到底是什么&#xff1f; 初学者很容易把微服务和分布式混为一谈&#xff0c;但其实二者之间存在非常大的差异&#xff0c;我个人认为主要有以下几点&#xff1a; 分布式主要是一种技术手段&#xff0c;用来保证多个相同的进程能够共同工作而不出错。采用各种复杂的…