javaagent获得每个方法的执行时间

比如说在对项目的每个方法的执行时间进行监控的需求的时候,不可能对项目中成千上万的方法一个一个的进行处理,当然如果时间要求很宽松的可以 一个一个的进行添加处理。

但是使用字节码插桩的方法,可以比较方便的进行需求的实现。

拦截的时候使用的是 javaagent
编辑、修改 class 字节码文件的时候使用的是 javassist 技术

在引入 javassist 的jar包的时候,尽可能的选择高版本的jar包


    javassist
    javassist
    3.12.1.GA

下面这个是,插桩需要的代码,将该java文件打包成jar包,

然后在项目启动的时候 添加 -javaagent:yyyy.jar=aaa.bbb.ccc

java -Dfile.encoding=utf-8 -jar xxxx.jar -javaagent:yyyy.jar=aaa.bbb.ccc

xxxx.jar     表示项目的jar包

yyyy.jar     表示插桩的jar包

aaa.bbb.ccc    要监控的类文件夹,或者类文件

具体到类文件的话,会打印出类中所有方法的执行时间

具体到类文件夹的话,会打印出类中所有方法的执行时间+所有类的总执行时间(总时间实在最后才会打印出来)

/**
 * 监控所有的方法类
 *
 * Create by yang_zzu on 2020/6/14 on 20:35
 */
public class PublicAgentMain {

    //javaagent 入口方法

    // 以 arg 为前缀的类才会进行插桩处理 -javaagent:xxx.jar=com.sys.insertPile
    public static void premain(String arg, Instrumentation instrumentation) {

        System.out.println("hello agent!!!!!");

        final String config = arg;

        // 使用 javassist ,在运行时修改 class 字节码,就是 插桩
        final ClassPool pool = new ClassPool();
        pool.appendSystemPath();

        instrumentation.addTransformer(new ClassFileTransformer() {
            @Override
            public byte[] transform(ClassLoader loader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {

                if (className == null || !className.replaceAll("/",".").startsWith(config)) {
                    return null;
                }

                try {
                    className = className.replaceAll("/", ".");
                    CtClass ctClass = pool.get(className);
                    // 获得类中的所有方法
                    for (CtMethod declaredMethod : ctClass.getDeclaredMethods()) {
                        newMethod(declaredMethod);
                    }
                    return ctClass.toBytecode();
                } catch (Exception e) {
                    e.printStackTrace();
                }

                return null;
            }
        });


    }

    //复制原有的方法(类似于使用 agent )
    private static CtMethod newMethod(CtMethod oldMethod) {
        CtMethod copy = null;
        try {
            //1. 将方法进行复制
            copy = CtNewMethod.copy(oldMethod, oldMethod.getDeclaringClass(), null);
            //类似于使用动态代理
            copy.setName(oldMethod.getName() + "$agent");
            //类文件中添加 sayHello$agent 方法
            oldMethod.getDeclaringClass().addMethod(copy);

            //2. 改变原有的方法,将 原有的 sayHello 方法进行重写操作
            if (oldMethod.getReturnType().equals(CtClass.voidType)) {
                oldMethod.setBody(String.format(voidSource, oldMethod.getName()));
            } else {
                oldMethod.setBody(String.format(source, oldMethod.getName()));
            }
        } catch (CannotCompileException | NotFoundException e) {
            e.printStackTrace();
        }
        return copy;

    }

    /**
     * 参数的封装
     * $$ ======》 arg1, arg2, arg3
     * $1 ======》 arg1
     * $2 ======》 arg2
     * $3 ======》 arg3
     * $args ======》 Object[]
     */
    //有返回值得方法
    final static String source = "{ long begin = System.currentTimeMillis();\n" +
            "        Object result;\n" +
            "        try {\n" +
            "            result = ($w)%s$agent($$);\n" + //s% 将参数传递到下一个方法,然后使用 s% 传递的参数进行替换操作, $w 表示的是在进行return的时候会强制的进行类型转换
            "        } finally {\n" +
            "            long end = System.currentTimeMillis();\n" +
            "            System.out.println(end - begin);\n" +
            "        }\n" +
            "        return ($r) result;}";

    //没有返回值的方法
    final static String voidSource = "{long begin = System.currentTimeMillis();\n" +
            "        try {\n" +
            "            %s$agent($$);\n" +
            "        } finally {\n" +
            "            long end = System.currentTimeMillis();\n" +
            "            System.out.println(end - begin);\n" +
            "        }}";


}

 监听的类的文件夹

javaagent获得每个方法的执行时间_第1张图片

javaagent获得每个方法的执行时间_第2张图片

 

pom 文件夹,打包的配置


com.sys.insertPilepublic.PublicAgentMain


    4.0.0

    com.sys.yang
    mybatisYang
    1.0-SNAPSHOT


    
        
            javassist
            javassist
            3.12.1.GA
        


        
            junit
            junit
            4.12
        
        
            mysql
            mysql-connector-java
            8.0.17
            runtime
        
        
            org.mybatis
            mybatis
            3.5.1
        
        
            commons-logging
            commons-logging
            1.2
        
        
            log4j
            log4j
            1.2.17
        
        
            org.springframework
            spring-jdbc
            5.1.0.RELEASE
            
                
                    org.springframework
                    spring-jcl
                
            
        
        
            org.springframework
            spring-context
            5.1.0.RELEASE
            
                
                    org.springframework
                    spring-jcl
                
            
        
        
            org.mybatis
            mybatis-spring
            2.0.2
        

    


    

        
            
                org.apache.maven.plugins
                maven-jar-plugin
                2.2
                
                    
                        
                            ${project.name}
                            ${project.version}


                            
                            com.sys.insertPilepublic.PublicAgentMain
                            false
                            false
                        
                    
                    true
                
            
            
                org.apache.maven.plugins
                maven-compiler-plugin
                2.3.2
                
                    1.8
                    1.8
                    utf8
                
            
        

        
            
                src/main/java
                
                    **/*.xml
                
                true
            
            
                src/test/java
                
                    **/*.xml
                
                true
            
        
    



UserService.java 文件

/**
 *
 * 字节码插桩
 * 运行的时候需要附着于其他的jar 包,运行
 * java -jar xxx.jar -javaagent:xxxx.jar
 *
 * Create by yang_zzu on 2020/6/14 on 17:35
 */
public class UserService {

    public void sayHello(String s) throws InterruptedException {
        Thread.sleep(50);
        System.out.println("hello world!!! " + s);
    }


    public Integer sayHelloReturn(String s, Integer age) throws InterruptedException {
        Thread.sleep(100);
        System.out.println("hello world!!! " + s + "age = " + age);
        return age;
    }

    public String sayHelloReturnEvery(String name, Integer age, String phone) throws InterruptedException {
        Thread.sleep(200);
        return (name + " 的年龄是 " + age + " 电话是 " + phone);
    }


}

 

AgentTest.java 测试类

/**
 * Create by yang_zzu on 2020/6/14 on 17:58
 */
public class AgentTest {

    public static void main(String[] args) {
        UserService userService = new UserService();

        UserService2 userService2 = new UserService2();
        try {
            userService.sayHello("你好世界@@@@");
            System.out.println("----------------------------------");
            System.out.println("年龄" + userService.sayHelloReturn("萨瓦迪卡", 18));
            System.out.println("----------------------------------");
            System.out.println(userService.sayHelloReturnEvery("小当家", 13, "110110110"));

            System.out.println("+++++++++++++++++++++++++++++++++++");

            userService2.sayHello("你好我的朋友!!!!!");
            System.out.println("----------------------------------");

            System.out.println(userService2.sayHelloReturn("xiaohua", 21));
            System.out.println("----------------------------------");
            System.out.println(userService2.sayHelloReturnEvery("haha", 10, "120110"));
            System.out.println("----------------------------------");


        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }

}

项目的结构

javaagent获得每个方法的执行时间_第3张图片

本来是不想粘贴测试类的代码,因为粘上代码看着内容比较长,也没有多大的实际意义。

https://blog.csdn.net/yang_zzu/article/details/106750330

这个是另外一个,javaagent 的使用,打印 idea 的所有类文件。这个只是作为学习使用,并没有什么实际性的应用。在破解 idea 的时候应用的就是 javaagent 技术

 

 

你可能感兴趣的:(java开发)