作者:@houkai
本文为作者原创,转载请注明出处:
Spring简化Java的下一个理念:基于切面的声明式编程
3、应用切面
依赖注入的目的是让相互协作的组件保持松散耦合;而AOP编程允许你把遍布应用各处的功能分离出来形成可重用的组件。
AOP面向切面编程被定义为促使应用程序分离关注点的一项技术。系统由许多不同的组件组成,每个组件除了负责某一特定的功能,还要承担额外的职责,诸如日志、事务管理和安全等等的服务经常融入到自身的核心业务逻辑中去,这些服务统称为横向关注点,因为它们总是跨越系统的各个组件。
将这些代码分散到多个组件,会导致双重复杂性:
如果要修改关注点得逻辑,必须修改各个组件的相关实现。即使你把这些关注点抽象成一个独立的模块,其他模块只是调用它的方法,但方法的调用还是会重复出现在各个模块中(耦合);
组件代码会因为那些与自身核心业务无关的代码而变得混乱。
假设你需要使用吟游诗人这个服务类来记载骑士(BraveKnight)的所有事迹,建立Minstrel(吟游诗人)类。
?123456789101112131415161718192021222324252627282930313233343536373839404142434445464748package com.test.knights; public class Minstrel { public void singBeforeQuest() { System.out.println("Fa la la; The knight is so brave!"); } public void singAfterQuest() { System.out.println("Tee hee he; The brave knight do a quest"); }}//代码效果参考:http://www.ezhiqi.com/zx/art_1662.html package com.test.knights; public class BraveKnight implements Knight { private Quest quest; private Minstrel minstrel; public BraKnight(Quest quest, Minstrel minstrel) { this.quest = quest; this.minstrel = minstrel; } public void embrakOnQuest() throws QuestException { minstrel.singBeforeQuest(); quest.embrak(); minstrel.singAfterQuest(); }//代码效果参考:http://www.ezhiqi.com/bx/art_3253.html }//代码效果参考:http://www.ezhiqi.com/zx/art_3918.html<?xml version="1.0" encoding="UTF-8"?>
让我们做适当的调整来让BraveKnight来使用Minstrel
这样达到了预期的效果:骑士做任务前执行singBeforeQuest()方法、任务后执行singAfterQuest()方法。但是管理吟游诗人真的是骑士的责任吗?骑士在依赖注入时就必须提供一个吟游诗人,这显然是不合逻辑的。吟游诗人应该自己独立出来,他有自己分内的事情。
改进,把Minstrel抽象为一个切面,你所做是是在Spring配置文件中声明它。
这里将minstrel Bean声明为一个切面,pointcut定义了一类切入点embark,expression含义是在任意返回类型、任意对象调用方法、任意入参的embrakOnQuest 方法都有效。
称为前置通知,称为后置通知。从这个示例中获得两个重要的观点:
Minstrel仍然是一个POJO,没有任何代码表示它将被作为一个切面使用。
Minstrel可以被应用到BraveKnight中,而BraveKnight不需要显示地调用。实际上,BraveKnight完全不知道Minstrel的存在。