java怎么加注解_JavaAgent技术之添加注解-程序员宅基地

技术标签: java怎么加注解  

什么是JavaAgent(Java探针)?你可以理解为Java版AOP。只不过这个AOP项目启动时运行一次

JavaAgent 只在项目启动时运行一次并且是java文件编译成class文件后才运行。所以不会影响到class文件。

JavaAgent 是寄生项目。即需要依赖一个正常项目才能运行

我这里演示使用 IDE 为 IDEA ,项目为maven普通项目结构,可以很方便的帮助我们创建一个 JavaAgent 项目。

按住CTRL+SHIFT+ALT+S 键,进入Project Structure界面

选择添加JAR

9eb1fd38c97a3410ce85a4ced226d349.png

选择项目,注意下面的链接要在SRC下。

e828fbff0de40bd7e0c6d23ebfdee506.png

创建好的目录结构,其中Premain-Class 是我后期添加上去的。这里指定的为后面的 Agent 类中的 premain 方法。同时也是JavaAgent的启动方法,Class-path中的值也是我们在maven中添加的包。你们应该没有

7d3c0033f68dbfebb67ec6d66f28a925.png

编写一个JavaAgent方法

package com.annie;

import java.lang.instrument.Instrumentation;

public class Agent {

// JavaAgent启动时调用的方法

public static void premain(String args, Instrumentation instrumentation) {

System.out.println("传入的参数为: " + args);

System.out.println("JavaAgent启动了...");

}

}

到这里一个JavaAgent项目就基本启动完成了。我们在创建一个普通项目(项目类型无所谓)

这里我创建的是一个SpringBoot的项目,因为比较好演示。

先将将JavaAgent编译jar包,编译好的jar包就在项目同级目录下的out下面。

ef72622b9183ea6c681eaae2b5cdb50e.png

b35f2e4b1fdc3ca7fbb29522340cb0fe.png

设置启动配置,后面一个Hello是传入的参数。可有可无。不要时不要忘记连 = 号也去掉。

-javaagent:C:\Users\13100\Documents\IdeaProjects\BaseJava\out\artifacts\JavaAgent_jar\JavaAgent.jar=Hello

保存运行

8cd8ac5ac1e8e14ee69792e4b59bc2b9.png

到这里一个JavaAgent项目就基本搭建成功了。当然我们不能满足于此。这里我自演示基本的动态注解其实以下就没有和JaveAgent相关内容了。

添加javassist包。

org.javassist

javassist

3.25.0-GA

编写代码(写完不要忘记重新编译)

package com.annie;

import javassist.ClassPool;

import javassist.CtClass;

import javassist.bytecode.AnnotationsAttribute;

import javassist.bytecode.AttributeInfo;

import javassist.bytecode.ClassFile;

import javassist.bytecode.ConstPool;

import javassist.bytecode.annotation.Annotation;

import java.io.ByteArrayInputStream;

import java.io.IOException;

import java.lang.instrument.ClassFileTransformer;

import java.lang.instrument.IllegalClassFormatException;

import java.lang.instrument.Instrumentation;

import java.security.ProtectionDomain;

import java.util.List;

public class Agent {

// JavaAgent启动时调用的方法

public static void premain(String args, Instrumentation instrumentation) {

System.out.println("传入的参数为: " + args);

System.out.println("JavaAgent启动了...");

// instrumentation 中包含了项目中的全部类。每加载一次.class文件就运行一次

instrumentation.addTransformer(new ClassFileTransformer() {

@Override

public byte[] transform(ClassLoader loader, String className, Class> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {

// 这里我只捕捉自己写 TestController 类

if(className.equals("com/annie/controller/TestController")){

// 接下来就是javassist的使用我会粗略的写下备注。不会详解。

// 获取一个 class 池。

ClassPool classPool = ClassPool.getDefault();

try {

// 创建一个新的 class 类。classfileBuffer 就是当前class的字节码

CtClass ctClass = classPool.makeClass(new ByteArrayInputStream(classfileBuffer));

ClassFile classFile = ctClass.getClassFile();

ConstPool constPool = classFile.getConstPool();

// 从这里取出原本类中的注解 建议DEBUG看下attributes中的数据

List attributes = classFile.getAttributes();

AnnotationsAttribute attributeInfo = (AnnotationsAttribute) attributes.get(1);

// 导包是javassist的包

// 添加新的注解

Annotation annotation = new Annotation("org.springframework.web.bind.annotation.RequestMapping", constPool);

attributeInfo.addAnnotation(annotation);

// 返回新的字节码

return ctClass.toBytecode();

} catch (Exception e) {

e.printStackTrace();

}

}

return new byte[0];

}

});

}

}

编写一个 TestController,这里我只给了RestController的注解。并没有给RequestMapping注解。RequestMapping由JavaAgent为我添加。

package com.annie.controller;

import org.springframework.web.bind.annotation.RequestMapping;

import org.springframework.web.bind.annotation.RestController;

import java.lang.annotation.Annotation;

@RestController

public class TestController {

@RequestMapping("/test")

public String test() {

// 利用反射查看Class注解。

Class extends TestController> aClass = this.getClass();

Annotation[] declaredAnnotations = aClass.getDeclaredAnnotations();

StringBuffer sb = new StringBuffer();

// 查看类上全部注解

for (Annotation annotation : declaredAnnotations) {

sb.append(annotation.toString() + "\n");

}

return sb.toString();

}

}

浏览器上输入 127.0.0.1:8080/test 查看TestController上的注解

c2055685fcc5fc2131877a557164c1f2.png

补充内容 linux 上发布项目

将javaAgent.jar 和 javaassist.jar 放入到一个文件夹中。

238255f099fb9aaee5cbf6583ee30b7f.png

运行命令

java -javaagent:JavaAgent.jar -jar web-test-0.0.1-SNAPSHOT.jar

项目就启动成功。注意一点这个jar包就是你bulid出来的JAR

标签:JavaAgent,java,jar,添加,import,注解,annotation,javassist

来源: https://blog.csdn.net/weixin_39933264/article/details/100181397

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_29066781/article/details/114522932

智能推荐

while循环&CPU占用率高问题深入分析与解决方案_main函数使用while(1)循环cpu占用99-程序员宅基地

文章浏览阅读3.8k次,点赞9次,收藏28次。直接上一个工作中碰到的问题,另外一个系统开启多线程调用我这边的接口,然后我这边会开启多线程批量查询第三方接口并且返回给调用方。使用的是两三年前别人遗留下来的方法,放到线上后发现确实是可以正常取到结果,但是一旦调用,CPU占用就直接100%(部署环境是win server服务器)。因此查看了下相关的老代码并使用JProfiler查看发现是在某个while循环的时候有问题。具体项目代码就不贴了,类似于下面这段代码。​​​​​​while(flag) {//your code;}这里的flag._main函数使用while(1)循环cpu占用99

【无标题】jetbrains idea shift f6不生效_idea shift +f6快捷键不生效-程序员宅基地

文章浏览阅读347次。idea shift f6 快捷键无效_idea shift +f6快捷键不生效

node.js学习笔记之Node中的核心模块_node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是-程序员宅基地

文章浏览阅读135次。Ecmacript 中没有DOM 和 BOM核心模块Node为JavaScript提供了很多服务器级别,这些API绝大多数都被包装到了一个具名和核心模块中了,例如文件操作的 fs 核心模块 ,http服务构建的http 模块 path 路径操作模块 os 操作系统信息模块// 用来获取机器信息的var os = require('os')// 用来操作路径的var path = require('path')// 获取当前机器的 CPU 信息console.log(os.cpus._node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是

数学建模【SPSS 下载-安装、方差分析与回归分析的SPSS实现(软件概述、方差分析、回归分析)】_化工数学模型数据回归软件-程序员宅基地

文章浏览阅读10w+次,点赞435次,收藏3.4k次。SPSS 22 下载安装过程7.6 方差分析与回归分析的SPSS实现7.6.1 SPSS软件概述1 SPSS版本与安装2 SPSS界面3 SPSS特点4 SPSS数据7.6.2 SPSS与方差分析1 单因素方差分析2 双因素方差分析7.6.3 SPSS与回归分析SPSS回归分析过程牙膏价格问题的回归分析_化工数学模型数据回归软件

利用hutool实现邮件发送功能_hutool发送邮件-程序员宅基地

文章浏览阅读7.5k次。如何利用hutool工具包实现邮件发送功能呢?1、首先引入hutool依赖<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.7.19</version></dependency>2、编写邮件发送工具类package com.pc.c..._hutool发送邮件

docker安装elasticsearch,elasticsearch-head,kibana,ik分词器_docker安装kibana连接elasticsearch并且elasticsearch有密码-程序员宅基地

文章浏览阅读867次,点赞2次,收藏2次。docker安装elasticsearch,elasticsearch-head,kibana,ik分词器安装方式基本有两种,一种是pull的方式,一种是Dockerfile的方式,由于pull的方式pull下来后还需配置许多东西且不便于复用,个人比较喜欢使用Dockerfile的方式所有docker支持的镜像基本都在https://hub.docker.com/docker的官网上能找到合..._docker安装kibana连接elasticsearch并且elasticsearch有密码

随便推点

Python 攻克移动开发失败!_beeware-程序员宅基地

文章浏览阅读1.3w次,点赞57次,收藏92次。整理 | 郑丽媛出品 | CSDN(ID:CSDNnews)近年来,随着机器学习的兴起,有一门编程语言逐渐变得火热——Python。得益于其针对机器学习提供了大量开源框架和第三方模块,内置..._beeware

Swift4.0_Timer 的基本使用_swift timer 暂停-程序员宅基地

文章浏览阅读7.9k次。//// ViewController.swift// Day_10_Timer//// Created by dongqiangfei on 2018/10/15.// Copyright 2018年 飞飞. All rights reserved.//import UIKitclass ViewController: UIViewController { ..._swift timer 暂停

元素三大等待-程序员宅基地

文章浏览阅读986次,点赞2次,收藏2次。1.硬性等待让当前线程暂停执行,应用场景:代码执行速度太快了,但是UI元素没有立马加载出来,造成两者不同步,这时候就可以让代码等待一下,再去执行找元素的动作线程休眠,强制等待 Thread.sleep(long mills)package com.example.demo;import org.junit.jupiter.api.Test;import org.openqa.selenium.By;import org.openqa.selenium.firefox.Firefox.._元素三大等待

Java软件工程师职位分析_java岗位分析-程序员宅基地

文章浏览阅读3k次,点赞4次,收藏14次。Java软件工程师职位分析_java岗位分析

Java:Unreachable code的解决方法_java unreachable code-程序员宅基地

文章浏览阅读2k次。Java:Unreachable code的解决方法_java unreachable code

标签data-*自定义属性值和根据data属性值查找对应标签_如何根据data-*属性获取对应的标签对象-程序员宅基地

文章浏览阅读1w次。1、html中设置标签data-*的值 标题 11111 222222、点击获取当前标签的data-url的值$('dd').on('click', function() { var urlVal = $(this).data('ur_如何根据data-*属性获取对应的标签对象

推荐文章

热门文章

相关标签