介绍@H_502_2@
@H_502_2@你以前听说过AOP(Aspect Oriented Programming)吗?虽然在PHP方面,好像没有过多的使用,但是在企业级开发中,AOP被广泛使用。我将借此文,向大家介绍PHP方面的AOP。这篇文章主要解释AOP的概念。@H_502_2@什么是AOP?@H_502_2@
@H_502_2@在应用开发中,我们经常发现需要很多功能,这些功能需要经常被分散在代码中的多个点上,但是这些点事实上跟实际业务没有任何关联。比如,在执行一些特殊任务之前需要确保用户是在登陆状态中,我们把这些特殊人物就叫做"cross-cutting concerns",让我们通过Wikipedia来了解一下"cross-cutting concerns"(横向关系)的定义。@H_502_2@在计算机科学中,"cross-cutting concerns"指的是“切面(或方向)编程”。这些关系不能从其他系统(框架设计或者某些实现)中很好的分解出来,以至于出现代码重复,在系统中存在有意义的依赖关系,或者两者兼有之。@H_502_2@现在你对于“横向关系”应该有一个基础的认识,让我们看看他们在代码中是怎么样的?假设一种场景,你是一个博客站点的编辑。你需要登陆站点,然后进行创建帖子,验证帖子,编辑帖子等等。如果你没有登陆,那么你应该直接到登陆界面。为了确保这些行为是安全的,以上的任何操作都需要进行有效验证,代码如下。@H_502_2@public function approvePost() {@H_502_2@ if (!Authentication::checkAuthentication()) {@H_502_2@ // redirect to login@H_502_2@ }@H_502_2@ else {@H_502_2@ // proceed@H_502_2@ }@H_502_2@ }
public function editPost() {@H_502_2@ if (!Authentication::checkAuthentication()) {@H_502_2@ // redirect to login@H_502_2@ }@H_502_2@ else {@H_502_2@ // proceed@H_502_2@ }@H_502_2@ }
public function viewPost() {@H_502_2@ // ...@H_502_2@ }@H_502_2@}@H_502_2@
单一职责原则讲述的是每个类应该只有单一的责任(任务),而且应该把整个责任都封装在一个类中。所有服务应该按照职责严谨而均衡的进行分布。
迄今为止,我们能够明白AOP所表达的意思。横向切面关系被成组的放进一个类中,我们管这个类叫“切面”。从我们核心代码中分离横向切面关系的过程就叫做Aspect Oriented Programming。AOP专业术语有很多条件专门用于解释AOP的特性。理解这些条件将是你成功把AOP集成到你的项目中的钥匙。@H_502_2@Aspect@H_502_2@Advice@H_502_2@Joinpoint@H_502_2@Pointcut@H_502_2@我们已经学习到切面(Aspect)是什么!现在让我们了解一下其他三个条件意味着什么?
Advice(通知) @H_502_2@
Advice用于调用Aspect(切面),正如其名所暗示,Advice用于定义某种情况下做什么和什么时间做这件事情。在我们之前的例子中,checkAuthentication(做什么)是advice(通知),在指定方法中它应该在执行代码之前(什么时间)被调用。 @H_502_2@Joinpoint(接入点) @H_502_2@
Joinpoint是我们创建Advice应用中的位置。再翻看之前的代码,你会发现我调用了几个与业务逻辑没有直接关联的功能。在createPost()中,如,cross-cutting concerns应该在执行验证逻辑之前和发送信息给管理员之后发生。这些都可能是接入点。在你的应用代码中,接入点可以放置在任何位置。但是Advice仅能在某些点中布置,这要根据你的AOP框架,过后我会讨论。Pointcut(点切割) @H_502_2@
点切割定义了一种把通知匹配到某些接入点的方式。虽然在我们的例子中只有一对接入点,但是在你的应用中你可以放置上千个接入点,你也不需要把通知应用到所有的接入点上。你可以把一些你认为有必要的接入点绑定到通知上。 假设我们想要通知 createPost(),approvePost() 和 editPost(),但是现在没有viewPost()。我们使用某种方法把这三种方法绑定到通知上。之后我们创建一个包含切面细节的XML文件,这些细节包含一些匹配接入点的正则表达式。 总结:当有横向切入关系存在于我们的应用的时候,我们可以创建一个切面,这个切面在一些选择使用点切割的接入点上应用通知功能。@H_502_2@AOP 通知类型@H_502_2@
@H_502_2@ 通知代码我们可以用很多中方式表现。我之前提到,这些通知代码依赖你使用的框架,但是有些你需要熟悉的类型,请看下面:@H_502_2@ 前通知@H_502_2@ 返回后通知@H_502_2@ 抛出后通知@H_502_2@ 周边通知前通知 @H_502_2@
在你的代码中一些特殊点之前使用通知——正常是调用一个方法。迄今为止,为了简化概念和为了让你更快的理解你的代码,我经常把通知写到方法里。但是在真实的环境里,通知经常是不写在方法里的。应该有一个独立的控制器,每个方法都在这个控制器里,而且每个方法都包裹着AOP的功能。这个全局的控制器运行在整个系统里,而且对我们是不可见的。@H_502_2@返回后通知 @H_502_2@
这个通知在指定功能执行完后只执行一次,并且返回那个访问点。考虑下面的代码:@H_502_2@@H_502_2@