JavaAgent 与 动态代理

ASM, CGlib, Java Proxy, Javassist都是可以操作字节码,但是这些操作字节码都需要等到类加载到JVM中之后再对字节码进行重写。
JavaAgent则是一个可以在加载前就进行重写,然后再加载的方式。

JavaAgent

JavaAgent 是JDK 1.5 以后引入的,也可以叫做Java代理。是运行在 main方法之前的拦截器,它内定的方法名叫 premain ,也就是说先执行 premain 方法然后再执行 main 方法。
那么如何实现一个 JavaAgent 呢?很简单,只需要增加 premain 方法即可。

public class TestPreMain {
    public static void premain(String agentOps, Instrumentation inst) {
        System.out.println("=========premain方法执行========");
        System.out.println(agentOps);
        inst.addTransformer(new FirstAgent());
    }
}

说明一下premain方法。该方法的原理与 main 应用程序入口点类似。在 Java 虚拟机 (JVM) 初始化后,每个 premain 方法将按照指定代理的顺序调用,然后将调用实际的应用程序 main 方法。

JVMTI(Java Virtual Machine Tool Interface)是一套本地编程接口集合,它提供了一套”代理”程序机制,可以支持第三方工具程序以代理的方式连接和访问 JVM,并利用 JVMTI 提供的丰富的编程接口,完成很多跟 JVM 相关的功能。关于 JVMTI 的详细信息,请参考 Java SE 6 文档(请参见 参考资源)当中的介绍。

java.lang.instrument 包的实现,也就是基于这种机制的:在 Instrumentation 的实现当中,存在一个 JVMTI 的代理程序,通过调用 JVMTI 当中 Java 类相关的函数来完成 Java 类的动态操作。

Instrumentation 的最大作用,就是类定义动态改变和操作。在 Java SE 5 及其后续版本当中,开发者可以在一个普通 Java 程序(带有 main 函数的 Java 类)运行时,通过 – javaagent参数指定一个特定的 jar 文件(包含 Instrumentation 代理)来启动 Instrumentation 的代理程序。


public class FirstAgent implements ClassFileTransformer {
    public final String injectedClassName = "test.agent.wxl";

    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {
        className = className.replace("/", ".");
//        System.out.println(className);
        if (className.startsWith(injectedClassName)) {
            CtClass ctclass = null;
            try {
                ctclass = ClassPool.getDefault().get(className);// 使用全称,用于取得字节码类<使用javassist>
                CtMethod[] ctmethods = ctclass.getMethods();
                for (CtMethod ctMethod : ctmethods) {
                    CodeAttribute ca = ctMethod.getMethodInfo2().getCodeAttribute();
                    if (ca == null) {
                        continue;
                    }
                    if (!ctMethod.isEmpty()) {
//                        System.out.println(ctMethod.getName());
                          ctMethod.insertBefore("System.out.println(\"hello Im agent : " + ctMethod.getName() + "\");");
                    }
                }
                return ctclass.toBytecode();
            } catch (Exception e) {
                System.out.println(e.getMessage());
                e.printStackTrace();
            }
        }
        return null;
    }
}

实现 ClassFileTransformer 这个接口的目的就是在class被装载到JVM之前将class字节码转换掉,从而达到动态注入代码的目的。其中代码的修改使用到了javassist。
上段代码实现了把特定包下面的class的所有类的所有有body的非native的方法前面添加一句
System.out.println("hello im agent :" + ctMethod.getName());

在pom.xml文件中添加插件:

          <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-shade-plugin</artifactId>
                <version>3.0.0</version>
                <executions>
                    <execution>
                        <phase>package</phase>
                        <goals>
                            <goal>shade</goal>
                        </goals>
                        <configuration>
                            <transformers>
                                <transformer
                                        implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                                    <manifestEntries>
                                        <Premain-Class>agent.TestPreMain</Premain-Class>
                                    </manifestEntries>
                                </transformer>
                            </transformers>
                        </configuration>
                    </execution>
                </executions>
            </plugin>

构建后会发现jar包里面的MANIFEST.MF文件如下

Manifest-Version: 1.0
Premain-Class: agent.TestPreMain
Archiver-Version: Plexus Archiver
Built-By: xiaolong.wei
Created-By: Apache Maven 3.3.9
Build-Jdk: 1.8.0_121

最后创建一个被代理的工程类

public class TestAgentDemo {

    public static void main(String... args) {
        testAgent();

        testAgent1("1", "2");
    }

    public static void testAgent() {
        System.out.println("test agent say hello");
    }

    public static void testAgent1(String one, String two) {
        System.out.println("test agent say hello" + one + two);
    }
}

在启动的时候使用-javaagent:来指定jar包

javaagent

运行:

运行结果

可以看到方法main, testAgent testAgent1 的前面都被加入了代码.

JavaAgent 实际应用中

1. 可以在加载java文件之前做拦截把字节码做修改
2. 获取所有已经被加载过的类
3. 获取某个对象的大小
4. 将某个jar加入到bootstrapclasspath里作为高优先级被bootstrapClassloader加载
5. 将某个jar加入到classpath里供AppClassload去加载
6. 设置某些native方法的前缀,主要在查找native方法的时候做规则匹配

动态代理

动态代理的实现:

  1. cglib (Code Generation Library) 实现
  2. InvocationHandler

代理模式

代理模式就是给某一个对象提供一个代理对象,并由代理对象控制对原对象的引用。通俗的来讲代理模式就是我们生活中常见的中介。

如下图

代理模式

实际使用中。客户端获取到的是代理对象,而代理对象ProxyObject把请求转发给RealObject,ProxyObject是不做业务的,实际工作的其实只有RealObject 。

如果编写一个类的同时就要一个代理类,那这样系统就会变得臃肿且难以维护,所以出现了动态代理。在运行状态中,需要代理的地方,根据AbstractObject和RealObject,动态地创建一个Proxy,用完之后,就会销毁,这样就可以避免了Proxy 角色的class在系统中冗杂的问题了。

InvocationHandler

在考虑代理类的时候,我们无非是想在代理类调用实际对象的前后进行一些业务处理,如下图。


代理

这就造成了代理类的通用性,只需要在invoke的前后加入特定的代码就可以了。所以就出现了将所有通用的调用真实方法的管理器,让这个触发管理器统一去管理就叫做Invocation Handler。

JDK通过 java.lang.reflect包下面的ProxyInvocationHandler来支持动态代理。

RealObject这个类创建一个动态代理对象,JDK主要会做以下工作:

1.   获取 RealObject上的所有接口列表;
2.   确定要生成的代理类的类名,默认为:com.sun.proxy.$ProxyXXXX ;
3.   根据需要实现的接口信息,在代码中动态创建 该Proxy类的字节码;
4 .  将对应的字节码转换为对应的class 对象;
5.   创建InvocationHandler实例handler,用来处理Proxy所有方法调用;
6.   Proxy 的class对象 以创建的handler对象为参数,实例化一个proxy对象

InvocationHandler,我们需要实现下列的invoke方法:
在调用代理对象中的每一个方法时,在代码内部,都是直接调用了InvocationHandler 的invoke方法,而invoke方法根据代理类传递给自己的method参数来区分是什么方法。

public interface Subject {
    int getInt(Integer i);
}
public interface Subject2 {
    String getString(String source);
}
public class RealSubject implements Subject, Subject2 {
    @Override
    public int getInt(Integer i) {
        System.out.println("this is method of getInt " + i++);
        return i;
    }

    @Override
    public String getString(String source) {
        System.out.println("source string is :" + source);
        source = source + " suffix";
        return source;
    }
}

代理类

public class SubjectProxy implements InvocationHandler {

    private RealSubject subject;

    public SubjectProxy(RealSubject subject) {
        this.subject = subject;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("method " + method.getName() + "invoke by proxy, start");
        Object o = method.invoke(subject, args);
        System.out.println("method " + method.getName() + "invoke by proxy, end");
        return o;
    }
}

测试代码

public class ProxyMain {
    public static void main(String... args) {
        RealSubject subject = new RealSubject();
        //创建一个Invoke Handler 
        SubjectProxy proxy = new SubjectProxy(subject);
        // 通过Proxy newProxyInstance方法创建代理类
        Object o = Proxy.newProxyInstance(ProxyMain.class.getClassLoader(), subject.getClass().getInterfaces(), proxy);
        Subject realSubject = (Subject) o;
        int result = realSubject.getInt(12);
        System.out.println("result:" + result);
        Subject2 s2 = (Subject2) o;
        s2.getString("hello");
    }
}

看一下代码运行结果


image.png

当然JDK的InvokeHandler也是生成了一个代理类。

JDK提供了sun.misc.ProxyGenerator.generateProxyClass(String proxyName,class[] interfaces) 底层方法来产生动态代理类的字节码:

cglib —— 通过类继承

JDK提供的生成动态代理的方式有个明显的特点就是被代理的类它必须要有实现的接口才行。既RealObject必须实现了某个接口或者某几个接口,才能代理接口这些接口的方法。
还好有CGLib。CGLIB(Code Generation Library),是一个强大的,高性能,高质量的Code生成类库,它可以在运行期扩展Java类与实现Java接口。而CGLIB底层使用的是ASM库。

cglib 创建某个类A的动态代理类的模式是:

1.   查找A上的所有非final 的public类型的方法定义;
2.   将这些方法的定义转换成字节码;
3.   将组成的字节码转换成相应的代理的class对象;
4.   实现 MethodInterceptor接口,用来处理 对代理类上所有方法的请求(这个接口和JDK动态代理InvocationHandler的功能和角色是一样的)

被代理类

public class ElectricCar {
    public boolean charging(Double fee) {
        System.out.println("charging ~~ Fee:" + fee);
        return true;
    }

    public boolean driver(String van, Integer color) {
        System.out.println(van + " is runing ");
        return true;
    }
}

代理方法

public class CarInterceptor implements MethodInterceptor {
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("intercept in car before");
        Object result = proxy.invokeSuper(obj, args);
        System.out.println("intercept in car after");
        return result;
    }
}

测试方法

public class CgLibMain {

    public static void main(String... args) {
        //生成class文件 保存
        System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\work\\agentDemo\\target");
      
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(ElectricCar.class);
        CarInterceptor carInterceptor = new CarInterceptor();
        //设置代理前后的Interceptor
        enhancer.setCallback(carInterceptor);
        ElectricCar electricCar = (ElectricCar) enhancer.create();
        electricCar.driver("川A12131", 1);
        electricCar.charging(12d);
    }
}

代码运行结果:


image.png

参考:
https://blog.csdn.net/luanlouis/article/details/24589193
https://www.ibm.com/developerworks/cn/java/j-lo-jse61/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • https://blog.csdn.net/luanlouis/article/details/24589193 ...
    小陈阿飞阅读 856评论 1 1
  • 一、基本概念 1.什么是代理? 在阐述JDK动态代理之前,我们很有必要先来弄明白代理的概念。代理这个词本身并不是计...
    小李弹花阅读 16,437评论 2 40
  • 思考:尊重自己的感觉,就是把一切交给身体,也就是感受来做,这比头脑下决定要舒服很多。 太过于理性的东西,给人一种过...
    杨雪雪阅读 183评论 0 0
  • 打造品牌,也许并不是每天企业正在做的事情,但必然是企业未来的发展方向,对于打造品牌各个企业也有自己的见解,也在根据...
    housheng阅读 417评论 0 0
  • 早已经习惯了孤独的活着,喜欢一个人安安静静的完成所有事情,喜欢独处,我想是因为找不到完全志同道合的朋友吧! 越发现...
    LoveQMT阅读 150评论 0 1