标签:
转载自:http://www.cnblogs.com/sunwke/articles/2568875.html
网上出现了很多讲解 AspectJ 的资料,但大多是从讲解 AspectJ 语法开始,然后讲解如何应用 AspectJ,如何分离软件开发过程的不同方面(Aspect)--Log,Session,Authentication and Authorization,Transaction,等等。
初次接触 AspectJ 的读者看到这些资料(或者语法手册),会感到 AspectJ 有些神秘。他们想知道,AspectJ 是如何做到这些的? AspectJ 是怎样工作的? AspectJ 需要特殊的运行环境吗?
本文从另一个角度讲解 AspectJ,本文从讲解 AspectJ 的设计思路、运行原理入手,回答上述问题。
本文讲解的主要内容,按照概念的重要程度,排列如下:
节简单介绍 AOP 的概念,解释我们为什么需要 AOP。
AOP 是 Object Oriented Programming(OOP)的补充。
OOP 能够很好地解决对象的数据和封装的问题,却不能很好的解决 Aspect("方面")分离的问题。下面举例具体说明。
比如,我们有一个 Bank(银行)类。Bank 有两个方法,deposit(存钱)和 withdraw(取钱)。
类和方法的定义如下:
Code 2.1 Bank.java class Bank{ public float deposit(AccountInfo account, float money){ // 增加 account 账户的钱数,返回账户里当前的钱数 } public float withdraw(AccountInfo account, float money){ // 减少 account 账户的钱数,返回取出的钱数 } }; |
这两个方法涉及到用户的账户资金等重要信息,必须要非常小心,所以编写完上面的商业逻辑之后,项目负责人又提出了新的要求 -- 给 Bank 类的每个重要方法加上安全认证特性。
于是,我们不得不分别在上面的两个方法中加入安全认证的代码。
类和方法的定义如下:(新增加的代码用不同的背景标出)
Code 2.2 Bank.java class Bank{ public float deposit(AccountInfo account, float money){ // 验证 account 是否为合法用户 // 增加 account 账户的钱数,返回账户里当前的钱数 } public float withdraw(AccountInfo account, float money){ // 验证 account 是否为合法用户 // 减少 account 账户的钱数,返回取出的钱数 } }; |
这两个方法都需要操作数据库,为了保持数据完整性,项目负责人又提出了新的要求 -- 给 Bank 类的每个操作数据库的方法加上事务控制。
于是,我们不得不分别在上面的两个方法中加入安全认证的代码。
类和方法的定义如下:
Code 2.3 Bank.java class Bank{ public float deposit(AccountInfo account, float money){ // 验证 account 是否为合法用户 // Begin Transaction // 增加 account 账户的钱数,返回账户里当前的钱数 // End Transaction } public float withdraw(AccountInfo account, float money){ // 验证 account 是否为合法用户 // Begin Transaction // 减少 account 账户的钱数,返回取出的钱数 // End Transaction } }; |
我们看到,这些与商业逻辑无关的重复代码遍布在整个程序中。实际的工程项目中涉及到的类和函数,远远不止两个。如何解决这种问题?
我们首先来看看 OOP 能否解决这个问题。
我们利用 Design Pattern 的 Template Pattern,可以抽出一个框架,改变上面的例子的整个设计结构。
类和方法的定义如下:
Code 2.4 Base.java abstract class Base{ public float importantMethod(AccountInfo account, float money){ // 验证 account 是否为合法用户 // Begin Transaction float result = yourBusiness(account, money) // End Transaction return result; } protected abstract float yourBusiness(AccountInfo account, float money); }; Code 2.5 BankDeposit.java class BankDeposit extends Base{ protected float yourBusiness(AccountInfo account, float money){ // 增加 account 账户的钱数,返回账户里当前的钱数 } }; Code 2.6 BankWithdraw.java class BankWithdraw extends Base{ protected float yourBusiness(AccountInfo account, float money){ // 减少 account 账户的钱数,返回取出的钱数 } }; |
这里我们用一种很勉强的方法实现了认证和事务代码的重用。而且,有心的读者可能会注意到,这种方法的前提是,强制所有的方法都遵守同样的 signature。
如果有一个转账方法 transfer(AccountInfo giver, AccountInfo receiver, float money),由于 transfer 方法的 signature 不同于 yourBusiness 的 signature,这个方法无法使用上面的框架。
这个例子中提到的认证,事务等方面,就是 AOP 所关心的 Aspect。
AOP 就是为了解决这种问题而出现的。AOP 的目的就是 --Separation of Aspects (or Separation of Concerns).
使用 AspectJ,我们不用对原有的代码做任何修改,就可以为代码提供不同的 Aspect(方面)-- 比如,认证,事务等。
我们只需要提供两个不同的 Aspect-- 认证 Aspect 和事务 Aspect。
Code 4.1 AuthAspect.java aspect AuthAspect{ pointcut bankMethods() : execution (* Bank.deposit( … )) || execution (* Bank. withdraw ( … )); Object around(): bankMethods(){ // 验证 account 是否为合法用户 return proceed(); } }; Code 4.2 TransactionAspect.java aspect TransactionAspect{ pointcut bankMethods() : execution(* Bank.deposit( … )) || execution (* Bank. withdraw ( … )); Object around(): bankMethods(){ // Begin Transaction Object result = proceed(); // End Transaction return result; } }; |
如果您暂时不能理解这段代码,没有关系,后面会讲到,这些 aspect 的定义,不过是定义了一些代码生成规则。
我们用 AspectJ 编译器编译 Bank 文件和含有 aspect 的这个文件,出来的结果就是带有安全认证和事务处理的 Bank 类。编译出来的这个 Bank 类调用了 AspectJ Runtime Lib,所以,如果你要运行这个 Bank 类,你需要把 AspectJ Runtime Lib 设置在你的 classpath 里面。
我们来看看,AspectJ 编译器为我们做了什么事情。
如何验证这一点?您可以到 http://www.eclipse.org/aspectj/下载安装 AspectJ,编译里面的 Sample,把编译结果反编译一下,就可以看到 AspetJ 自动生成的代码。
我们看到,AspectJ 是一种代码自动生成工具。你编写一段通用的代码,比如认证方面的代码,事务方面的代码,然后根据 AspectJ 语法定义一套代码生成规则(aspect 定义),AspectJ 就会帮助你自动把这段通用代码分布到对应的代码里面去,简单快捷.
转载自:http://www.cnblogs.com/sunwke/articles/2568875.html
标签:
原文地址:http://www.cnblogs.com/SummerinShire/p/5441862.html