面向切面编程(AOP)的理解


Posted in Javascript onMay 01, 2015

在传统的编写业务逻辑处理代码时,我们通常会习惯性地做几件事情:日志记录、事务控制及权限控制等,然后才是编写核心的业务逻辑处理代码。当代码编写完成回头再看时,不禁发现,扬扬洒洒上百行代码中,真正用于核心业务逻辑处理才那么几行,如图6-4所示。方法复方法,类复类,就这样子带着无可奈何遗憾地度过了多少个春秋。这倒也罢,倘若到了项目的尾声,突然决定在权限控制上需要进行大的变动时,成千上万个方法又得一一"登门拜访",痛苦"雪上加霜"。

如果能把图6-4中众多方法中的所有共有代码全部抽取出来,放置到某个地方集中管理,然后在具体运行时,再由容器动态织入这些共有代码的话,最起码可以解决两个问题:

Java EE程序员在编写具体的业务逻辑处理方法时,只需关心核心的业务逻辑处理,既提高了工作效率,又使代码变更简洁优雅。

在日后的维护中由于业务逻辑代码与共有代码分开存放,而且共有代码是集中存放的,因此使维护工作变得简单轻松。

面向切面编程AOP技术就是为解决这个问题而诞生的,切面就是横切面,如图6-5所示,代表的是一个普遍存在的共有功能,例如,日志切面、权限切面及事务切面等。

下面我们以用户管理业务逻辑组件UserService的AOP实现过程(见图6-6)为例,深度剖析一下AOP技术的实现原理。AOP技术是建立在Java语言的反射机制与动态代理机制之上的。业务逻辑组件在运行过程中,AOP容器会动态创建一个代理对象供使用者调用,该代理对象已经按Java EE程序员的意图将切面成功切入到目标方法的连接点上,从而使切面的功能与业务逻辑的功能同时得以执行。从原理上讲,调用者直接调用的其实是AOP容器动态生成的代理对象,再由代理对象调用目标对象完成原始的业务逻辑处理,而代理对象则已经将切面与业务逻辑方法进行了合成。

现将图6-6中涉及到的一些概念解释如下。

切面(Aspect):其实就是共有功能的实现。如日志切面、权限切面、事务切面等。在实际应用中通常是一个存放共有功能实现的普通Java类,之所以能被AOP容器识别成切面,是在配置中指定的。

通知(Advice):是切面的具体实现。以目标方法为参照点,根据放置的地方不同,可分为前置通知(Before)、后置通知(AfterReturning)、异常通知(AfterThrowing)、最终通知(After)与环绕通知(Around)5种。在实际应用中通常是切面类中的一个方法,具体属于哪类通知,同样是在配置中指定的。

连接点(Joinpoint):就是程序在运行过程中能够插入切面的地点。例如,方法调用、异常抛出或字段修改等,但Spring只支持方法级的连接点。

切入点(Pointcut):用于定义通知应该切入到哪些连接点上。不同的通知通常需要切入到不同的连接点上,这种精准的匹配是由切入点的正则表达式来定义的。

目标对象(Target):就是那些即将切入切面的对象,也就是那些被通知的对象。这些对象中已经只剩下干干净净的核心业务逻辑代码了,所有的共有功能代码等待AOP容器的切入。

代理对象(Proxy):将通知应用到目标对象之后被动态创建的对象。可以简单地理解为,代理对象的功能等于目标对象的核心业务逻辑功能加上共有功能。代理对象对于使用者而言是透明的,是程序运行过程中的产物。

织入(Weaving):将切面应用到目标对象从而创建一个新的代理对象的过程。这个过程可以发生在编译期、类装载期及运行期,当然不同的发生点有着不同的前提条件。譬如发生在编译期的话,就要求有一个支持这种AOP实现的特殊编译器;发生在类装载期,就要求有一个支持AOP实现的特殊类装载器;只有发生在运行期,则可直接通过Java语言的反射机制与动态代理机制来动态实现。

以下是补充:

AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。

AOP与OOP是面向不同领域的两种设计思想。

OOP(面向对象编程)针对业务处理过程的实体及其属性和行为进行抽象封装,以获得更加清晰高效的逻辑单元划分。

AOP则是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果。

我们可以单单从上面的字面上来理解AOP和OOP的话,用下面的理解也不为过:

OOP实际上是对对象的属性和行为的封装,而AOP对于这点就无从谈起,但是AOP是处理某个步骤和阶段的,从中进行切面的提取,也就是说,如果几个或更多个逻辑过程中,有重复的操作行为,AOP就可以提取出来,运用动态代理,实现程序功能的统一维护,这么说来可能太含蓄,如果说到权限判断,日志记录等,可能就明白了。如果我们单纯使用OOP,那么权限判断怎么办?在每个操作前都加入权限判断?日志记录怎么办?在每个方法里的开始、结束、异常的地方手动添加日志?所有,如果使用AOP就可以借助代理完成这些重复的操作,就能够在逻辑过程中,降低各部分之间的耦合了。二者扬长补短,互相结合最好。

下面详细了解一些AOP的概念:

•方面(Aspect):一个关注点的模块化,这个关注点实现可能另外横切多个对象。事务管理是J2EE应用中一个很好的横切关注点例子。方面用Spring的Advisor或拦截器实现。
•连接点(Joinpoint):程序执行过程中明确的点,如方法的调用或特定的异常被抛出。
•通知(Advice):在特定的连接点,AOP框架执行的动作。各种类型的通知包括“around”、“before”和“throws”通知。通知类型将在下面讨论。许多AOP框架包括Spring都是以拦截器做通知模型,维护一个“围绕”连接点的拦截器链。
•切入点(Pointcut):指定一个通知将被引发的一系列连接点的集合。AOP框架必须允许开发者指定切入点,例如,使用正则表达式。
•引入(Introduction):添加方法或字段到被通知的类。Spring允许引入新的接口到任何被通知的对象。例如,你可以使用一个引入使任何对象实现IsModified接口,来简化缓存。
•目标对象(Target Object):包含连接点的对象,也被称作被通知或被代理对象。
•AOP代理(AOP Proxy):AOP框架创建的对象,包含通知。在Spring中,AOP代理可以是JDK动态代理或CGLIB代理。
•编织(Weaving):组装方面来创建一个被通知对象。这可以在编译时完成(例如使用AspectJ编译器),也可以在运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。

Spring 中 AOP 代理由 Spring 的 IoC 容器负责生成、管理,其依赖关系也由 IoC 容器负责管理。至于Spring的AOP在项目中具体是怎么实现的,下篇博客会以日志记录为例进行学习。

Javascript 相关文章推荐
jQuery.extend 函数详解
Feb 03 Javascript
JavaScript中Cookies的相关使用教程
Jun 04 Javascript
JavaScript Split()方法
Dec 18 Javascript
jQuery中的siblings用法实例分析
Dec 24 Javascript
AngularJS实现数据列表的增加、删除和上移下移等功能实例
Sep 05 Javascript
AngularJS中$watch和$timeout的使用示例
Sep 20 Javascript
iOS和Android用同一个二维码实现跳转下载链接的方法
Sep 28 Javascript
使用jsonp实现跨域获取数据实例讲解
Dec 25 Javascript
解决vue router使用 history 模式刷新后404问题
Jul 19 Javascript
Bootstrap一款超好用的前端框架
Sep 25 Javascript
深入浅出vue图片路径的实现
Sep 04 Javascript
如何实现echarts markline标签名显示自己想要的
Jul 20 Javascript
yui3的AOP(面向切面编程)和OOP(面向对象编程)
May 01 #Javascript
使用AOP改善javascript代码
May 01 #Javascript
Javascript aop(面向切面编程)之around(环绕)分析
May 01 #Javascript
jQuery插件Zclip实现完美兼容个浏览器点击复制内容到剪贴板
Apr 30 #Javascript
jQuery插件slider实现拖动滑块选取价格范围
Apr 30 #Javascript
javascript实现验证身份证号的有效性并提示
Apr 30 #Javascript
PHP+jQuery实现随意拖动层并即时保存拖动位置
Apr 30 #Javascript
You might like
Zerg兵种介绍
2020/03/14 星际争霸
php curl 登录163邮箱并抓取邮箱好友列表的代码(经测试)
2011/04/07 PHP
yii2 开发api接口时优雅的处理全局异常的方法
2019/05/14 PHP
重写javascript中window.confirm的行为
2012/10/21 Javascript
jquery向上向下取整适合分页查询
2014/09/06 Javascript
基于javascript代码实现通过点击图片显示原图片
2015/11/29 Javascript
JavaScript原生数组Array常用方法
2017/04/06 Javascript
JavaScript数据结构之数组的表示方法示例
2017/04/12 Javascript
JS库之Waypoints的用法详解
2017/09/13 Javascript
微信小程序picker组件下拉框选择input输入框的实例
2017/09/20 Javascript
nuxt+axios解决前后端分离SSR的示例代码
2017/10/24 Javascript
angular2 ng2-file-upload上传示例代码
2018/08/23 Javascript
Vue绑定内联样式问题
2018/10/17 Javascript
解决iview多表头动态更改列元素发生的错误的方法
2018/11/02 Javascript
浅谈JS和jQuery的区别
2019/03/27 jQuery
前端插件之Bootstrap Dual Listbox使用教程
2019/07/23 Javascript
vue设置导航栏、侧边栏为公共页面的例子
2019/11/01 Javascript
Vue自定义组件双向绑定实现原理及方法详解
2020/09/03 Javascript
vue中的计算属性和侦听属性
2020/11/06 Javascript
跟老齐学Python之再深点,更懂list
2014/09/20 Python
python解决字符串倒序输出的问题
2018/06/25 Python
Python+pandas计算数据相关系数的实例
2018/07/03 Python
浅谈pycharm使用及设置方法
2019/09/09 Python
Python对wav文件的重采样实例
2020/02/25 Python
HTML块级标签汇总(小篇)
2016/07/13 HTML / CSS
美国奢侈品购物平台:Orchard Mile
2018/05/02 全球购物
国贸类专业毕业生的求职信分享
2013/12/08 职场文书
蜜蜂引路教学反思
2014/02/04 职场文书
保研推荐信
2014/05/09 职场文书
人事专员岗位说明书
2014/07/29 职场文书
2015年护士节活动策划方案
2015/05/04 职场文书
python实现简单倒计时功能
2021/04/21 Python
python神经网络编程之手写数字识别
2021/05/08 Python
用Python实现一个打字速度测试工具来测试你的手速
2021/05/28 Python
PO模式在selenium自动化测试框架的优势
2022/03/20 Python
Tomcat starup.bat 脚本实现开机自启动
2022/04/20 Servers