@[TOC](用最通俗的方法讲spring [一] ──── AOP)

写这个系列的目的(可以跳过不看)

自己写这个系列的目的,是因为自己是个比较笨的人,我曾一度怀疑自己的智商不适合干编程这个行业.因为在我自学的过程中,看过无数别人的文章,博客,心得.可那一大堆的术语,技术基础,根本就难以阅读理解.再加上网上文章良莠不齐,很多文章都是写给已经懂的人看的.一个例子:Q:什么是控制反转.A:将对在自身对象中的一个内置对象的控制反转,反转后不再由自己本身的对象进行控制这个内置对象的创建,而是由第三方系统去控制这个内置对象的创建.???这种专业的解释,在我看来就不是给学习的人看的,而是给已经有相当经验的人读的.难道不能用更加感性的解释这些东西吗?编程语言作为一门语言,我认为一定程度上是相当感性的东西,既然我们互相说话时可以听懂,那么,这些技术就可以在一定程度上,完全可以用感性的方式解释.

AOP

AOP是面向切面.得,说了白说.什么是面向切面?这是个问题,但我们先不着急回答.

一. 举个例子

我们的代码结构,大多都是MVC模式.那我们用MVC举个简单的例子.MVC模式的话,最常用的spring + springMVC + Mybatis,大多数是下面的样子.

┌───────────────┐
|   view        | 用户界面
├───────────────┤
|   Controller  | 控制层
├───────────────┤
|   Service     | 服务层,也叫业务层
├───────────────┤
|   Dao         | 持久层,也叫数据访问层
├───────────────┤
|   Database    | 数据库
└───────────────┘

外加Model作为数据载体在各个层之间传来传去只要是接触过JavaWeb开发的,上面都看的懂,如果这个不懂暂时还是不要深究这些知识,先以框架应用为主.

这个Service是个UserService,作用只有一个:保存用户.

┌────────────────────┐
|   view             |
├────────────────────┤
|   Controller       |
├────────────────────┤
|   UserService.add  | ───> 新增用户
├────────────────────┤
|   Dao              |
├────────────────────┤
|   Database         |
└────────────────────┘

现在,有需求要我们在Service中增加日志,开发这个功能的同事不在了,经理要求我们来做这个需求.那么根据面向对象的理念.调用service方法前要调用一个日志方法,调用完后要调用一个日志方法,那么代码中就要增加两行代码.于是,代码成了这样:

┌────────────────────┐
|   view             |
├────────────────────┤
|   Controller       |
├────────────────────┤
|   log.info...      | ───> 调用日志
|   UserService.add  | ───> 新增用户
|   log.info...      | ───> 调用日志
├────────────────────┤
|   Dao              |
├────────────────────┤
|   Database         |
└────────────────────┘

这么写正确吗?答案是当然正确,需求解决了,日志正常保存了,领导很高兴.

第二天,又有需求要控制service的事务,OK,然后代码变成了这样.

┌────────────────────┐
|   view             |
├────────────────────┤
|   Controller       |
├────────────────────┤
|   Transaction      | ───> 事务管理
|   log.info...      | ───> 调用日志
|   UserService.add  | ───> 新增用户
|   log.info...      | ───> 调用日志
|   Transaction      | ───> 事务管理
├────────────────────┤
|   Dao              |
├────────────────────┤
|   Database         |
└────────────────────┘

第三天,项目经理要求我们记录方法的运行所用时间,小意思,然后代码变成了这样

┌────────────────────┐
|   view             |
├────────────────────┤
|   Controller       |
├────────────────────┤
|   beginTime        | ───> 开始时间
|   Transaction      | ───> 事务管理
|   log.info...      | ───> 调用日志
|   UserService.add  | ───> 新增用户
|   log.info...      | ───> 调用日志
|   Transaction      | ───> 事务管理
|   endTime          | ───> 结束时间
├────────────────────┤
|   Dao              |
├────────────────────┤
|   Database         |
└────────────────────┘

第四天,客户要求我们实现新增用户时的权限校验,好说,代码变成了这样

┌────────────────────┐
|   view             |
├────────────────────┤
|   Controller       |
├────────────────────┤
|   permissions      | ───> 校验权限
|   beginTime        | ───> 开始时间
|   Transaction      | ───> 事务管理
|   log.info...      | ───> 调用日志
|   UserService.add  | ───> 新增用户
|   log.info...      | ───> 调用日志
|   Transaction      | ───> 事务管理
|   endTime          | ───> 结束时间
├────────────────────┤
|   Dao              |
├────────────────────┤
|   Database         |
└────────────────────┘

/*
不知道你们怎么想,但我现在,仍然觉得这种写法挺好的.
通俗易懂,可读性强到不知道哪里去了.
当然有个前提,这个项目只有这一个方法.
*/

第五天,开发UserService的同事回来一看,一脸懵逼,明明走的时候只有一行,现在却有一大堆不知道什么意思的代码存在.得了,他在此基础上开发,增加各种方法,功能.随着需求的变动,慢慢的,一百个service都变成了上面那个样子.接着记录日志的功能有了变动,你发现会影响到调用的类,搜了一下整个项目,发现有特么几千个地方都调用了这个方法.你心中一万头草泥马掠过,然后提出了辞职.接着有个新人接手了这个项目,打开一看,这映入眼帘的一大坨屎一样的是特么什么东西??硬着头皮改了几版,每个版本都有BUG,领导一怒之下,项目作废了,重金找了几个人重新做了.

==那么问题来了==现在的需求搞砸了可以辞职,项目烂了可以重做.那世界上第一个遇到这个问题的人,是怎么解决的?技术就是这技术,重做不还是一样会遇到这些问题吗?那我还怎么拓展功能呢?难道所有项目最终都会走向这种绝境?于是前人开始探索:

┌───────────────┐
|   Controller  |
|               ┼───────> 拓展方法写在这?
├───────────────┤
|   Service     |
├───────────────┤
|               ┼───────> 拓展方法写在这?
|    Dao        |
└───────────────┘

这不都是一回事吗?根本不解决问题.

我等凡夫俗子解决不了,但世界上有的是自带外挂的人.于是,有那么几个人就找到了解决办法!既然我要拓展方法,就要在你的方法里写代码,那么我能不能在不改你代码的基础上来拓展呢?

==比如在这个地方!==

┌───────────────┐
|   Controller  |
├───────────────┼───────> 比如,写在这!
|   Service     |
└───────────────┘

what the hell ??????你是让我在一个莫须有的地方写代码?

当然不是,大神的意思是代码要这样写.

┌───────────────┐
|   Controller  |      ┌───────────────┐
├───────────────┼──────┤ log.info...   |
|   Service     |      └───────────────┘
└───────────────┘

但执行的时候,会是这样

┌───────────────┐
|   Controller  |
├───────────────┤
|   log.info... |
|   Service     |
└───────────────┘

what the hell ???????????????????????????????这是为什么?

==答案很简单:==首先你既然打算了解AOP,那你应该是有一些Java基础的.我们都知道Java文件是没有用的,这就相当于一个txt文件,当程序真正运行的时候,是把Java文件通过编译器编译成class文件来运行的.那么我们好像看到重点了,编译器!既然最终运行的是class文件,而class文件又是编译器生成的,那我们是不是可以在编译器上动动手脚呢?没错,想要达成上面的效果,就需要一个不同的编译器.把日志这段代码编译到service中当然了,这个编译器不需要我们开发,而是早有人搞定了.(若是我能开发,我早就去谷歌开发者大会上做演讲了= =)这就是大名鼎鼎的 :==ajc编译器==(他属于AspectJ框架,可以单独使用此框架写个例子,可以加深理解)

1. 想象一下,有这么两坨东西

┌────────────────────┐
|   Controller.java  |
├────────────────────┤           ┌─────────────────┐
|   Service.java     |           |  log.info.java  |
└────────────────────┘           └─────────────────┘

2. 然后开始编译 : 编译器开始把Service.java中的代码编织成一个class文件.
3. ajc编译器开始编译 : ajc开始把log.info.java编织插入到service中.

┌────────────────────┐
|   Controller.java  |    _________________
├────────────────────┤ _/ log.info.java   |
|   Service.java     |  \ ________________|
└────────────────────┘

4. 就像上面,把log.info织入到这个这个方法中.
5. 最终形成的calss文件,就是这样

┌───────────────┐
|   Controller  |
├───────────────┤
|   log.info... |
|   Service     |
└───────────────┘

成了,问题解决了.有了这个编译器,前面的问题有迎刃而解.==ps : ajc编译器并不是唯一解决办法==

那么问题又又来了,==面向切面是什么?==也许你已经有答案了.

┌───────────────┐
|   Controller  |        ┌──────────────────────────────────┐
├───────────────┼──────> | 我们在这个角度写代码,就是面向切面.  |
|   Service     |        |                                  |
├───────────────┼──────> | 我们在这个角度写代码,就是面向切面.  |
|      Dao      |        └──────────────────────────────────┘
└───────────────┘

我们站在那条线的角度写代码,就是面向切面.那条线,是不是将原本的代码分成了两个部分呢?更形象一点

		┌───────────────┐
		|   Controller  |
		└───────────────┘
───────────────────────────────────── 这条线,像不像把整个代码切开的一条线呢?那么这条线,就叫切面
───────────────────────────────────── 如 : 日志调用
───────────────────────────────────── 如 : 调用时间
───────────────────────────────────── 如 : 事务管理
		┌───────────────┐
		|   Service     |
		├───────────────┤
		|      Dao      |
		└───────────────┘

上面的每一个如,就是一个切面类,也相当于一个切面,毕竟没有人规定我们必须只有一个切面不是?

面向切面是一种人们面对难题时的解决方案,他并不是一个新的语言或是新的技术.他消除了面向对象的一系列编程陋习.或者说解决了面向对象的一些缺点.

到了这里,我相信有些人还是不太懂面向切面的意义或者作用是什么,没关系,不要气馁,你一定比我当时可聪明多了.请继续往下看.

	 这张图可能并不太能直观的表达面向切面意义有多么重大.
	 ┌───────────────┐
	 |   Controller  |
	 └───────────────┘
	 ─────────────────────────
	 ┌───────────────┐
	 |   Service     |
	 ├───────────────┤
	 |      Dao      |
	 └───────────────┘

	 那么,这张图呢?

     ┌────────────────┬────────────────┬────────────────┬────────────────┬────────────────┬────────────────┐
     |   Controller1  |   Controller2  |   Controller3  |   Controller4  |   Controller5  |   Controller6  |
     └────────────────┴────────────────┴────────────────┴────────────────┴────────────────┴────────────────┘
─────────────────────── 面向切面的方法,可以织入到所有的service中,而不需要改变service的代码 ─────────────────────────
─────────────────────── 权限校验 ───────────────────────────────────────────────────────────────────────────────
─────────────────────── 日志调用 ───────────────────────────────────────────────────────────────────────────────
─────────────────────── 调用时间 ───────────────────────────────────────────────────────────────────────────────
─────────────────────── 事务管理 ───────────────────────────────────────────────────────────────────────────────
     ┌────────────────┬────────────────┬────────────────┬────────────────┬────────────────┬────────────────┐
     |   Service1     |   Service2     |   Service3     |   Service4     |   Service5     |   Service6     |
     └────────────────┴────────────────┴────────────────┴────────────────┴────────────────┴────────────────┘
─────────────────────── 面向切面的方法,可以织入到所有的Dao中,而不需要改变Dao的代码 ─────────────────────────────────
─────────────────────── 日志调用 ───────────────────────────────────────────────────────────────────────────────
─────────────────────── 调用时间 ───────────────────────────────────────────────────────────────────────────────
     ┌────────────────┬────────────────┬────────────────┬────────────────┬────────────────┬────────────────┐
     |      Dao1      |      Dao2      |      Dao3      |      Dao4      |      Dao5      |      Dao6      |
     └────────────────┴────────────────┴────────────────┴────────────────┴────────────────┴────────────────┘

	 我只需要声明一个类,就可以该类中的方法插入到任意类的位置

有些人说,用代理模式就可以解决上面所说的编程时的一系列问题.没错!面向切面 ─── 是思想.代理模式 ─── 是思想的实现.

二. 画个图片

上面画过了

三. 写个代码

只讲概念,后面讲例子

四. 说个人话

切面的几大概念

1.切面 (Aspect) :

拓展的类概念 : Aspect 声明类似于Java中的类声明,在Aspect中会包含着一些切点以及相应的增强.人话 : 创建一个类,这个类中的<增强>将插入到<目标对象>代码中.

2.连接点 (Joint point) :

调用点概念 : 表示在程序中明确定义的点,典型的包括方法调用,对类成员的访问以及异常处理程序块的执行等等,它自身还可以嵌套其它 joint point.人话 : 被<切点>所包含的内容,如切点为 userService.add()方法 : 连接点就是userService.add()如切点为 com.xxx包下的所有service : com.xxx包下的所有service的任意一个方法如切点为 com.xxx包下的所有add开头的方法 : com.xxx包下的所有add开头的方法

3.切点 (Pointcut):

在哪里增强概念 : 表示一组<连接点>,这些 joint point 或是通过逻辑关系组合起来,或是通过通配、正则表达式等方式集中起来,它定义了相应的增强将要发生的地方.或是我们要织入的地方人话 : 要往哪里插入如 userService.add()方法或 com.xxx包下的所有service方法或 com.xxx包下的所有add开头的方法

4.增强 (Advice) :

拓展方法概念 : Advice 定义了在<切点>里面定义的程序点具体要做的操作,它通过 before、after 和 around 来区别是在每个<连接点>之前、之后还是代替执行的代码.人话 : 切面类中需要定义的,连接点方法调用前的拓展,之后的拓展,或者之前和之后的拓展.

5.目标对象 (Target) :

被增强的目标概念 : 增强(Advice) 的目标对象..人话 : 被增强的目标

6.织入 (Weaving):

插入的动作概念 : 将 Aspect 和其他对象连接起来, 并创建 Adviced object 的过程.人话 : 将切面类插入进去

例如:
我们要在service的add方法里里增加日志
┌───────────────┐
|   Controller  |
└───────────────┘
─────────────────────────────────────    <-- com.xxx.service.UService  ─── <切点>
┌───────────────┐      ┌────────────┐    <-- log.info所在的类           ─── <切面>
| UService.add  |      |  log.info  |    <-- log.info方法就是           ─── <增强>
├───────────────┤      └────────────┘
|      Dao      |
└───────────────┘

UService.add的调用就是   ─── <连接点>
UService                ─── <目标对象>

到这里你应该理解什么是面向切面了,如果你还是不理解,不怕,可能是我的表达方式不适合你,全网有很多对于面向对象的优秀理解.虽然筛选好文章需要时间,我可以帮你找出来,但其实在筛选的过程中,更是一个理解的过程.

08-25 00:16