设计模式——代理模式的思考

代理模式是一种通过中间代理访问目标对象,以达到增强或拓展原对象功能目的的设计模式,举个例子来说,我们在购买飞机票时往往会通过一些第三方平台来购买,在这里第三方平台就可看成代理对象,目标对象则是各大航空公司,常见的代理方式有静态代理、动态代理以及Cglib代理。

静态代理

静态代理属于比较典型的代理模式,它的类图如下所示,从图中可以看到客户端是通过代理类的接口来访问目标对象的接口,也就是目标对象和代理类是一一对应的,如果有多个目标接口需要代理则产生多个代理类,实现方式比较冗余,另外如果拓展接口,对应的目标对象和代理类也需修改,不易维护。


image

动态代理

动态代理通过Java反射机制或者ASM字节码技术,动态地在内存中构建代理对象,从而实现对目标对象的代理功能。它与静态代理的主要区别在与动态代理的代理类是在运行期才会生成的,也就是说不会在编译期代理类的Class文件。常见的动态代理有JDK动态代理和Cglib动态代理。

JDK动态代理

JDK动态代理又称接口代理,它要求目标对象必须实现接口,否则不能代理。动态代理是基于java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler类来实现的,其中Proxy是拦截发生的地方,而InvocationHandler则是发生调用地方,newProxyInstance方法返回一个指定接口的代理类实例。
newProxyInstance方法

public static Object newProxyInstance(ClassLoader loader,  //目标对象的类加载器
                                      Class<?>[] interfaces, // 目标对象所实现的接口
                                      InvocationHandler h) // 事件处理器

InvocationHandler的Invoke方法

public Object invoke(Object obj, Object... args) // 该方法会调用目标对象对应的方法

在这里抛出一个问题,JDK动态代理为什么必须实现接口才能代理?要弄明白这个问题,我们需要拿到生成的代理类,下面是通过技术手段拿到的运行期的代理类,可以看到$Proxy0代理类已经继承Proxy类,由于Java是单继承的,所以只能通过实现接口的方式来实现。

public final class $Proxy0 extends Proxy implements IUserDao {
    private static Method m1;
    private static Method m2;
    private static Method m0;
    private static Method m3;

    public $Proxy0(InvocationHandler var1) throws  {
        super(var1);
    }
    ...

    public final void register() throws  {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }
    ...
}

CGLIB代理

CGLib相对于JDK动态代理更加灵活,它是通过生成子类来拓展目标对象的功能,使用cglib代理的对象无需实现接口,可以做到代理类无侵入,另外因CGLib具备很好的性能,所以被很多AOP框架所引用,比如Spring、Hibernate。
Cglib代理方式是通过继承来实现,其中代理对象是由Enhancer创建(Enhancer是Cglib字节码增强器,可以很方便对类进行拓展),另外,可以通过实现MethodInterceptor接口来定义方法拦截器。

public Object getProxyInstance() {
    Enhancer en = new Enhancer();
    // 继承被代理类
    en.setSuperclass(target.getClass());
    // 设置回调函数
    en.setCallback(new MethodInterceptor() {
        @Override
        public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
            System.out.println("开启事务");
            // 执行目标对象的方法
            Object returnValue = method.invoke(target, objects);
            System.out.println("关闭事务");
            return null;
        }
    });
    return en.create();
}

UserDao$$EnhancerByCGLIB$$b0e8b18d是获取到的UserDao的Cglib代理,可以看到它继承了UserDao方法,并为UserDao的每个方法生成了2个代理方法(这里只保留了register方法),第一个代理方法CGLIB$register$0()是直接调用父类的方法,第二个方法register()是代理类真正调用的方法,它会判断是否实现了MethodInterceptor接口,如果实现就会调用intercept方法,MethodInterceptor即为setCallback时注入的MethodInterceptor的实现类。

public class UserDao$$EnhancerByCGLIB$$b0e8b18d extends UserDao implements Factory {
    ...
    final void CGLIB$register$0() {
        super.register();
    }

    public final void register() {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (this.CGLIB$CALLBACK_0 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }
        // 判断是否实现了MethodInterceptor接口
        if (var10000 != null) {
            var10000.intercept(this, CGLIB$register$0$Method, CGLIB$emptyArgs, CGLIB$register$0$Proxy);
        } else {
            super.register();
        }
    }
    ...
}

Spring AOP

Spring AOP是基于动态代理实现的对代码无侵入的代码增强方式,它从本质上来说,是将Spring生成代理类对象放入IOC容器中,每次获取目标对象bean时都是通过getBean()方法,如果一个类被代理,那么实际通过getBean方法获取的就是代理类的对象,这也是Spring AOP为什么只能作用于IOC容器中的对象。
Spring AOP默认使用的JDK动态代理,如果目标对象没有实现接口,才会使用CGLib来代理,当然也可以强制使用CGLib代理,只需加上@EnableAspectJAutoProxy(proxyTargetClass = true)注解,@EnableAspectJAutoProxy一般用来开启Aspect注解配置,如果是基于xml配置的,在配置文件添加<aop:aspectj-autoproxy/>即可。
org.aopalliance包下有两个核心接口,分别是MethodInvocationMethodInterceptor,这两个接口也是Spring AOP中的核心类

  • MethodInvocation: AOP对需要增强方法的封装,它是真正执行AOP拦截的,该接口只包含getMethod()方法。
  • MethodInterceptor:AOP方法拦截器,AOP的相关操作一般在其内部完成
    下面代码是JdkDynamicAopProxy类,它是Spring AOP中JDK动态代理的具体实现,其中invoke()方法作为代理对象的回调函数被触发,通过invoke方法具体实现来完成对目标对象方法调用拦截或者功能增强,在invoke()方法中会创建一个ReflectiveMethodInvocation对象,该对象的proceed()方法会调用下一个拦截器,直至拦截器链被调用结束。
final class JdkDynamicAopProxy implements AopProxy, InvocationHandler, Serializable {

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        MethodInvocation invocation;
        Object oldProxy = null;
        boolean setProxyContext = false;

        TargetSource targetSource = this.advised.targetSource;
        Object target = null;

        try {
            ...
            //获得定义好的拦截器链(增强处理)
            List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);
            //如果拦截器链为空,执行原方法
            if (chain.isEmpty()) {
                Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
                retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse);
            }
            else {
                            // ReflectiveMethodInvocation实现了ProxyMethodInvocation接口
                            // ProxyMethodInvocation继承自MethodInvocation     
                invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
                // 执行proceed方法,调用下一个拦截器,直至拦截器链被调用结束,拿到返回值
                retVal = invocation.proceed();
            }
            Class<?> returnType = method.getReturnType();
            if (retVal != null && retVal == target &&
                    returnType != Object.class && returnType.isInstance(proxy) &&
                    !RawTargetAccess.class.isAssignableFrom(method.getDeclaringClass())) {
                retVal = proxy;
            }
            else if (retVal == null && returnType != Void.TYPE && returnType.isPrimitive()) {
                throw new AopInvocationException(
                        "Null return value from advice does not match primitive return type for: " + method);
            }
            return retVal;
        }
        finally {
            if (target != null && !targetSource.isStatic()) {
                // Must have come from TargetSource.
                targetSource.releaseTarget(target);
            }
            if (setProxyContext) {
                // Restore old proxy.
                AopContext.setCurrentProxy(oldProxy);
            }
        }
    }
}

解决自我调用时无法增强的问题

TestProxyImpl被Spring Aop增强时,testA()方法内部调用tesB()方法,那么testB()也会被增强吗?实际是不会的,从下面的输出结果可以看到testB()方法未被增强,可以很容易想到testB()未被增强的根本原因是this指的目标对象而非代理类对象

@Component
public class TestProxyImpl implements ITestProxy {
    @Override
    public void testA() {
        System.out.println("testA() execute ...");
        this.testB();
    }

    @Override
    public void testB() {
        System.out.println("testB() execute ...");
    }
}
// 输出
[AOP] Before ...
testA() execute ...
testB() execute ...

如果想在testA()方法调用testB()方法时增强testB()方法,即实际调用代理对象的testB()方法,下面有两种方法可以做到。
设置expose-proxy属性为true
如果是Spring Boot项目可以直接使用@EnableAspectJAutoProxy(exposeProxy = true)来暴露代理对象,如果是使用XML配置的,则用<aop:config expose-proxy="true">配置即可。该方法的原理就是使用ThreadLocal暂存代理对象,然后通过AopContext.currentProxy()方法重新拿到代理对象。

// JdkDynamicAopProxy类invoke方法中的代码片段
// 判断expose-proxy属性是否true
if (this.advised.exposeProxy) {
    // 暂存到ThreadLocal中,可点入setCurrentProxy方法查看
    oldProxy = AopContext.setCurrentProxy(proxy);
    setProxyContext = true;
}

为了能拿到代理对象,可以testA()方法做如下修改

public void testA() {
        System.out.println("testA() execute ...");
        //从ThreadLocal中取出代理对象,前提已设置expose-proxy属性为true,暴露了代理对象
        ITestProxy proxy = (ITestProxy) AopContext.currentProxy();
        proxy.testB();
 }

获取代理对象的Bean
还有一种方式和上面方法的原理差不多,都是获取的代理对象再调用testB()方法,不过该方法直接从Spring容器中获取,下面直接贴代码了~

@Component(value = "testProxy")
public class TestProxyImpl implements ITestProxy,ApplicationContextAware {
    
    private ApplicationContext applicationContext;

    @Override
    public void testA() {
        System.out.println("testA() execute ...");
        applicationContext.getBean("testProxy", ITestProxy.class).testB();
    }

    @Override
    public void testB() {
        System.out.println("testB() execute ...");
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
}

本文相关代码地址:https://github.com/LJWLgl/java-demo/tree/master/design-patterns/

原文链接:https://blog.ganzhiqiang.wang/2019/02/17/设计模式——代理模式的思考/#more

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

推荐阅读更多精彩内容

  • Java设计模式——代理模式 代理模式主要分为接口,委托类,代理类 接口:规定具体方法委托类:实现接口,完成具体的...
    vczyh阅读 658评论 0 0
  • 一、概述   代理模式我们接触的就比较多了,所谓的代理模式就是,给某一个对象提供一个代理对象,并由代理对象控制对原...
    骑着乌龟去看海阅读 901评论 0 9
  • SpringAOP 博客链接 本文主要是解析Spring AOP的运作流程。上次讲到Java中的两种动态代理技术:...
    spilledyear阅读 1,685评论 2 32
  • 做好时间管理,不仅仅需要做好时间统计,还需要掌握必要的方法和原则。时间管理不单纯的每天做一份计划,更重要的计划完成...
    未央笔记RM阅读 271评论 0 0
  • 时间很快来到70年代末,整个新中国呈现一股新气象,全国到处都是水库、铁路各种工地。父亲也参与了三线建设。举全国之力...
    笔行者阅读 308评论 0 2