Java 代理模式与 AOP

本文首发于 https://jaychen.cc
作者 jaychen

最近在学 Spring,研究了下 AOP 和代理模式,写点心得和大家分享下。

AOP

先说下AOP,AOP 全称 Aspect Oriented Programming,面向切面编程,和 OOP 一样也是一种编程思想。AOP 出现的原因是为了解决 OOP 在处理 侵入性业务上的不足。

那么,什么是侵入性业务?类似日志统计、性能分析等就属于侵入性业务。本来原本的业务逻辑代码优雅大气,正常运行,突然说需要在这段逻辑里面加上性能分析,于是代码就变成了下面这个样子


long begin = System.currentTimeMillis(); 

// 原本的业务
doSomething();

long end = System.currentTimeMillis();
long step = end - begin;
System.out.println("执行花费 :" + step);

从上面的代码看到,性能分析的业务代码和原本的业务代码混在了一起,好端端的代码就这么被糟蹋了。所以,侵入性业务必须有一个更好的解决方案,这个解决方案就是 AOP。

那么,AOP 是如何解决这类问题?

代理模式

通常,我们会使用代理模式来实现 AOP,这就意味着代理模式可以优雅的解决侵入性业务问题。所以下面来重点分析下代理模式。

这个是代理模式的类图。很多人可能看不懂类图,但是说实话有时候一图胜千言,这里稍微解释下类图的含义,尤其是类图中存在的几种连线符。

  • 矩形代表一个类,矩形内部的信息有:类名,属性和方法。
  • 虚线 + 三角空心箭头为 is=a 的关系,表示继承,所以上图中 TestSQLPerformance 都实现 IDatabase 接口。
  • 实线 + 箭头为关联关系,一般在代码中以成员变量的形式体现,所以上图中 Performance 类有一个 TestSQL 的成员变量。

有了类图,我们可以根据类图直接写出代理模式的代码了。这里代理模式分为静态代理和动态代理两种,我们分别来看下。

静态代理

假设一个场景,我们需要测试一条 sql query 执行所花费的时间。

如果按照普通的方式,代码逻辑应该如下

long begin = System.currentTimeMillis(); 

query();

long end = System.currentTimeMillis();
long step = end - begin;
System.out.println("执行花费 :" + step);

上面说过了,这种会导致查询逻辑和性能测试逻辑混淆在一块,那么来看看使用代理模式是如何解决这个问题的。

代理模式,代理,意味着有一方代替另一方完成一件事。这里,我们会编写两个类:TestSQL 为query 执行逻辑,Performance 为性能测试类。这里 Performance 会代替 TestSQL 去执行 query 逻辑。

要想 Performance 能够代替 TestSQL 执行 query 逻辑,那么这两个类应该是有血缘关系的,即这两个必须实现同一个接口。

// 接口
public interface IDatabase {
    void query();
}


public class TestSQL implements IDatabase {

    @Override
    public void query() {
        System.out.println("执行 query。。。。");
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


// 代理类
public class PerformanceMonitor implements IDatabase {
    TestSQL sql;

    public PerformanceMonitor(TestSQL sql) {
        this.sql = sql;
    }


    @Override
    public void query() {
        long begin = System.currentTimeMillis();

        // 业务逻辑。
        sql.query();

        long end = System.currentTimeMillis();
        long step = end - begin;
        System.out.println("执行花费 : " + step);
    }
}


// 测试代码
public class Main {
    public static void main(String[] strings) {
        TestSQL sql = new TestSQL();

        PerformanceMonitor performanceMonitor = new PerformanceMonitor(sql);
        // 由 Performance 代替 testSQL 执行
        performanceMonitor.query();
    }
}

从上面的示例代码可以分析出来代理模式是如何运作的,这里我们可以很明显看出代理模式的优越性,TestSQL 的逻辑很纯粹,没有混入其他无关的业务代码。

动态代理

回顾静态代理的代码,发现代理类 Performance 必须实现 IDatabase 接口。如果有很多业务需要用到代理来实现,那么每个业务都需要定义一个代理类,这会导致类迅速膨胀,为了避免这点,Java 提供了动态代理。

为何称之为动态代理,动态代理底层是使用反射实现的,是在程序运行期间动态的创建接口的实现。在静态代理中,我们需要在编码的时候编写 Performance 类实现 IDatabase 接口。而使用动态代理,我们不必编写 Performance 实现 IDatabase 接口,而是 JDK 在底层通过反射技术动态创建一个 IDatabase 接口的实现。

使用动态代理需要使用到 InvocationHandlerProxy 这两个类。

// 代理类,不再实现 IDatabase 接口,而是实现 InvocationHandler 接口
public class Performance implements InvocationHandler {

    private TestSQL sql;

    public Performance(TestSQL sql) {
        this.sql = sql;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        long begin = System.currentTimeMillis();

        // method.invoke 实际上就是调用 sql.query()
        Object object = method.invoke(sql, args);

        long end = System.currentTimeMillis();
        long step = end - begin;
        System.out.println("执行花费 :" + step);
        return object;
    }
}



public class Main {
    public static void main(String[] strings) {
        TestSQL sql = new TestSQL();
        Performance performance = new Performance(sql);

        IDatabase proxy = (IDatabase) Proxy.newProxyInstance(
                sql.getClass().getClassLoader(),
                sql.getClass().getInterfaces(),
                performance
        );
        proxy.query();
    }
}

先来看看 newProxyInstance 函数,这个函数的作用就是用来动态创建一个代理对象的类,这个函数需要三个参数:

  • 第一个参数为类加载器,如果不懂是什么玩意,先套着模板写,等我写下一篇文章拯救你。
  • 第二个参数为要代理的接口,在这个例子里面就是 IDatabase 接口。
  • 第三个参数为实现 InvocationHandler 接口的对象。

执行 newProxyInstance 之后,Java 会在底层自动生成一个代理类,其代码大概如下:

public final class $Proxy1 extends Proxy implements IDatabase{
    private InvocationHandler h;

    private $Proxy1(){}

    public $Proxy1(InvocationHandler h){
        this.h = h;
    }

    public void query(){
        ////创建method对象
        Method method = Subject.class.getMethod("query");
        //调用了invoke方法
        h.invoke(this, method, new Object[]{}); 
    }
}

你会发现,这个类很像在静态代理中的 Performance 类,是的,动态代理其本质是 Java 自动为我们生成了一个 $Proxy1 代理类。在 mian 函数中 newProxyInstance 的返回值就是该类的一个实例。并且,$Proxy1 中的 h 属性就是 newProxyInstance 的第三个参数。所以,当我们在 main 函数中执行 proxy.query(),实际上是调用 $proxy1#query 方法,进而再调用 Performance#invoke 方法。而在 Performance#invoke 通过 Object object = method.invoke(sql, args); 调用了 TestSQL#query 方法。

回顾上面的流程,理解动态代理的核心在于理解 Java 自动生成的代理类。这里还有一点要说明,JDK 的动态代理有一个不足:它只能为接口创建代理实例。这句话体现在代码上就是 newProxyInstance 的第二个参数是一个接口数组。为什么会存在这个不足?其实看 $Proxy1 代理类就知道了,这个由 JDK 生成的代理类需要继承 Proxy 类,而 Java 只支持单继承,所以就限制了 JDK 的动态代理只能为接口创建代理。

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

推荐阅读更多精彩内容