Java-Agent 原理-统计方法调用时间 | 学习笔记

简介: 快速学习 Java-Agent 原理-统计方法调用时间

开发者学堂课程【分布式链路追踪 SkywalkingJava-Agent 原理-统计方法调用时间】学习笔记,与课程紧密连接,让用户快速学习知识。

课程地址https://developer.aliyun.com/learning/course/743/detail/13166


Java-Agent 原理-统计方法调用时间


Java-Agent 原理-统计方法调用时间

上节课学习了 Java-Agent-环境搭建,这节课学习Skywaiking 如何对方法调用时间进行统计。

Skywaiking 利用 Java agent 和 ByteBuddy 技术来统计方法的调用时长,但Java agent无法统计方法内的调用时间,所以需要学习新的框架,ByteBuddy。

Bytebuddy 是开源的、基于 Apache2.0 许可证的库,主要目的是解决字节码操作和 instumentation API 的复杂性。对于一些字节码增强的代码操作变得更简单。

之前对方法调用时间进行统计是利用的 Spring 的 AUG 技术但现在的探针并没有集成 Spring 的框架所以无法利用该技术完成操作,因此要利用字节码对其增强,增强过后通过拦截器拦截达到统计方法时间的目的。

在此过程中就利用到了 ByteBuddy。下面就是 ButeBuddy如何实现的。

添加依赖

首先将下面的依赖添加到 pom 文件中

image.png

由于在打包时声明了 jar-with-dependencies 该参数,所以最终 ByteBuddy 的架包也会被打入到 java-agent-demo 的架包中。

修改 premain 的教学方法:

1、原先的教学方法代码:publicstatic void premain(string agentparam,Instumentation inst){

System.out.println(“premain执行1”);

System.out.println(agentparam);

}

2.现在的教学方法:利用 ByteBuddy 提供的 AgentBuilder.Transformer 类别操作。代码如下:

Transformer=new AgentBuilder.Tansformer(){

Public DynamicType.Builder<?>transform(DynamicType.Builder<?>builder,TypeDescription typeDescription

//method 指定哪些方法需要被拦截,ElementMatcher.any 指定了所有的方法

//声明 intercept 拦截器

return builder.method(ElementMatchers.<MethodDescription>any())

intercept(MethodDelegation.to(MyInterceptor.class));

}

};

//type 指定了 agent 拦截的包名,以 com.agent 作为前缀

//指定了 transformer

//将配置安装到 Instrumentation

new AgentBuilder.Default().type(ElementMatchers.<TypeDescription>nameStartswith(prefix:“com.agent”)).transform(transformer).installon(inst);

}

将 MyInterceptor 拷贝在 Java 目录下新建一个类,代码如下:

Import net.bytebuddy.implementation.bind.annotation.origin;

public class MyInterceptor{

@RuntimeType

public static object intercept(@origin Method method,

@supercall Callable<?>caallable){

Long start=System.currentTimeMillis();

Try{

Return callable.call();

}finally{

System.out.println(method.getName()+“:”+

(system.currentTimeMillis()-start)+“ms”);

}

}

3.对以上代码文件进行打包

image.png

打包过后修改 main 函数

4.首先要把包名改成

image.png

com.agent

5.修改 main 方法代码:

Pakage com.agent;Public static void main(string[]args)throws InterruptedException{

System.out.println(“main方法执行”)

Thread.sleep(millis:1000L);

}

}

6.执行

执行时间是由拦截器来拦截同时打印出来。

如果将包名替换拦截器就和原先设置的包名匹配不上,当执行 main 方法时就不会出现执行时间了。

如图:

image.png

说明 Java agent 和 ButeBuddy 组合起来就可以对方法的执行时间进行统计。

7.要点总结

Myinterceptor 拦截器是在 PreMainAgent 定义

transformer 转换器的时候指定的, Myintercepter 拦截器会对所有方法进行拦截,同时这些方法一定要存在于com.agent 这些包下才能生效。

修改包名时一定要将包名改为 com.agent。

执行的时候还要在 premain 方法里声明一个 transformer转

换器,同时将转换器安装到 instrumentation 中。

相关文章
|
5月前
|
存储 缓存 Java
我们来详细讲一讲 Java NIO 底层原理
我是小假 期待与你的下一次相遇 ~
206 2
|
4月前
|
监控 Java API
现代 Java IO 高性能实践从原理到落地的高效实现路径与实战指南
本文深入解析现代Java高性能IO实践,涵盖异步非阻塞IO、操作系统优化、大文件处理、响应式网络编程与数据库访问,结合Netty、Reactor等技术落地高并发应用,助力构建高效可扩展的IO系统。
150 0
|
5月前
|
Java API 微服务
2025 年 Java 从入门到精通学习笔记全新版
《Java学习笔记:从入门到精通(2025更新版)》是一本全面覆盖Java开发核心技能的指南,适合零基础到高级开发者。内容包括Java基础(如开发环境配置、核心语法增强)、面向对象编程(密封类、接口增强)、进阶技术(虚拟线程、结构化并发、向量API)、实用类库与框架(HTTP客户端、Spring Boot)、微服务与云原生(容器化、Kubernetes)、响应式编程(Reactor、WebFlux)、函数式编程(Stream API)、测试技术(JUnit 5、Mockito)、数据持久化(JPA、R2DBC)以及实战项目(Todo应用)。
356 5
|
2月前
|
小程序 Java 知识图谱
Java 学习笔记 —— BMI & BMR 计算器
这是一个使用 Java 编写的 BMI 与 BMR 计算器小程序,可输入年龄、性别、身高和体重,计算身体质量指数(BMI)和基础代谢率(BMR),并输出健康评估结果。通过该项目,掌握了 Java 的输入处理、数据验证、条件判断、数学运算及格式化输出等基础知识,是 Java 初学者的理想练习项目。
|
2月前
|
Java
Java 数组学习笔记
本文整理Java数组常用操作:遍历、求和、查找、最值及二维数组行求和等典型练习,涵盖静态初始化、元素翻倍、去极值求平均等实例,帮助掌握数组基础与应用。
|
6月前
|
存储 缓存 Java
【高薪程序员必看】万字长文拆解Java并发编程!(5):深入理解JMM:Java内存模型的三大特性与volatile底层原理
JMM,Java Memory Model,Java内存模型,定义了主内存,工作内存,确保Java在不同平台上的正确运行主内存Main Memory:所有线程共享的内存区域,所有的变量都存储在主存中工作内存Working Memory:每个线程拥有自己的工作内存,用于保存变量的副本.线程执行过程中先将主内存中的变量读到工作内存中,对变量进行操作之后再将变量写入主内存,jvm概念说明主内存所有线程共享的内存区域,存储原始变量(堆内存中的对象实例和静态变量)工作内存。
232 0
|
8月前
|
存储 Java
# 【Java全栈学习笔记-U1-day02】变量+数据类型+运算符
本篇笔记主要围绕Java全栈学习的第二天内容展开,涵盖了变量、数据类型、运算符以及Scanner类的应用。首先介绍了变量的概念与命名规范,以及如何定义和使用变量;接着详细讲解了Java中的基本数据类型,包括整型、浮点型、字符型、布尔型等,并通过实例演示了数据类型的运用。随后,深入探讨了各类运算符(赋值、算术、关系、逻辑)及其优先级,帮助理解表达式的构成。最后,介绍了如何利用Scanner类实现用户输入功能,并通过多个综合示例(如计算圆面积、购物打折、变量交换及银行利息计算)巩固所学知识。完成相关作业将进一步加深对这些基础概念的理解与实践能力。
158 13
|
5月前
|
存储 算法 安全
Java中的对称加密算法的原理与实现
本文详细解析了Java中三种常用对称加密算法(AES、DES、3DES)的实现原理及应用。对称加密使用相同密钥进行加解密,适合数据安全传输与存储。AES作为现代标准,支持128/192/256位密钥,安全性高;DES采用56位密钥,现已不够安全;3DES通过三重加密增强安全性,但性能较低。文章提供了各算法的具体Java代码示例,便于快速上手实现加密解密操作,帮助用户根据需求选择合适的加密方案保护数据安全。
416 58
|
4月前
|
人工智能 安全 Java
Go与Java泛型原理简介
本文介绍了Go与Java泛型的实现原理。Go通过单态化为不同类型生成函数副本,提升运行效率;而Java则采用类型擦除,将泛型转为Object类型处理,保持兼容性但牺牲部分类型安全。两种机制各有优劣,适用于不同场景。
169 24
|
5月前
|
XML JSON Java
Java 反射:从原理到实战的全面解析与应用指南
本文深度解析Java反射机制,从原理到实战应用全覆盖。首先讲解反射的概念与核心原理,包括类加载过程和`Class`对象的作用;接着详细分析反射的核心API用法,如`Class`、`Constructor`、`Method`和`Field`的操作方法;最后通过动态代理和注解驱动配置解析等实战场景,帮助读者掌握反射技术的实际应用。内容翔实,适合希望深入理解Java反射机制的开发者。
549 13