`

Aspect 学习

阅读更多
参照:王海龙 (buaawhl@sina.com)的 What is AspectJ(IBM developerWorks 中国 )


1. 序

Aspect Oriented Programming (AOP)是近来一个比较热门的话题。
AspectJ是AOP的Java语言的实现,获得了Java程序员的广泛关注。
关于AspectJ和AOP的具体资料,请从下列链接中查找:
http://www.eclipse.org/aspectj/
http://www.parc.com/research/csl/projects/aspectj/
http://aosd.net/

初次接触AspectJ的读者看到这些资料(或者语法手册),会感到AspectJ有些神秘。
他们想知道,AspectJ是如何做到这些的?AspectJ是怎样工作的?AspectJ需要特殊的运行环境吗?
本文从讲解AspectJ的设计思路、运行原理入手,回答上述问题。

本文讲解的主要内容,按照概念的重要程度,排列如下:

AspectJ是一个代码生成工具(Code Generator)。
AspectJ语法就是用来定义代码生成规则的语法。您如果使用过Java Compiler Compiler (JavaCC),您会发现,两者的代码生成规则的理念惊人相似。
AspectJ有自己的语法编译工具,编译的结果是Java Class文件,运行的时候,classpath需要包含AspectJ的一个jar文件(Runtime lib)。
AspectJ和xDoclet的比较。AspectJ和EJB Descriptor的比较。


2.Aspect Oriented Programming (AOP)

本节简单介绍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
}

}
;


我们看到,这些与商业逻辑无关的重复代码遍布在整个程序中。实际的工程项目中涉及到的类和函数,远远不止两个。如何解决这种问题?
这个例子中提到的认证,事务等方面,就是AOP所关心的Aspect。
AOP就是为了解决这种问题而出现的。AOP的目的就是--Separation of Aspects (or Separation of Concerns).
下面的章节,解释AspectJ工具如何解决Separation of Aspects的问题。


3.AspectJ

这一节我们来看看AspectJ如何实现上例中的Separation of Aspects。
使用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编译器为我们做了什么事情。

首先,AspectJ从文件列表里取出所有的文件名,然后读取这些文件,进行分析。
AspectJ发现一些文件含有aspect的定义,在这个例子里,就是AuthAspect和TransactionAspect的定义;这些aspect就是代码生成规则。
AspectJ根据这些aspect代码生成规则,修改添加你的源代码。在这个例子里,就是修改添加Bank文件。
AspectJ读取AuthAspect的定义,发现了一个pointcut--bankMethods();这个pointcut的定义是execution(* Bank.deposit(…)) || execution(* Bank. withdraw (…)),表示所有对Bank类的deposit和withdraw方法的执行点。
AspectJ继续读取AuthAspect的定义,发现了一个around(),这在AspectJ中叫做Advice,我不明白为什么叫这个名字,不过没关系,我们只要知道它是干什么的就行了。Advice允许你在某个类的方法的调用之前或调用之后,加入另外的代码。Code 4.1所示代码中的around()的" // 验证account是否为合法用户"部分,就是要加入的代码。这段代码要加在哪里呢?around()后面跟了一个pointcut--bankMethods()。根据这个pointcut,AspectJ会把这段代码加入到Bank.deposit和Bank.withdraw两个方法的执行之前。达到的效果就如同Code 2.2所示。
AspectJ读取TransactionAspect的定义,象第(4)步一样,发现了发现了一个pointcut--bankMethods()。
AspectJ继续读取AuthAspect的定义,发现了一个around()。这次AspectJ把"Begin Transaction"和"End Transaction"两段代码加在Bank.deposit和Bank. withdraw两个方法的执行前后。达到的效果就如同Code 2.3所示。

如何验证这一点?您可以到 http://www.eclipse.org/aspectj/下载安装AspectJ,编译里面的Sample,把编译结果反编译一下,就可以看到AspetJ自动生成的代码。
我们看到,AspectJ是一种代码自动生成工具。你编写一段通用的代码,比如认证方面的代码,事务方面的代码,然后根据AspectJ语法定义一套代码生成规则(aspect定义),AspectJ就会帮助你自动把这段通用代码分布到对应的代码里面去,简单快捷,算无遗策。
无独有偶,一个著名的编译器生成工具--Java Compiler Compiler (JavaCC),也采用了非常相似的代码生成机制。JavaCC允许你在语法定义规则文件中,加入你自己的Java代码,用来处理读入的各种语法元素。
AspectJ令你的代码更精简,结构更良好。AspectJ的好处,我就不多说了,网上很多精彩的文章探讨AspectJ的各种用途。

 

4.总结

开源项目的出现,打破了软件技术领域的众多壁垒,推动软件技术进程的日新月异。
同时,一些新名词,新概念也层出不穷,令人眼花缭乱,无所适从。其实,很多东西都是换汤不换药,我们理解应用这些新技术的时候,要抓住本质,要破除迷信,破除任何人为的神秘感。

互联网时代的权威,不是说出来的,而是做出来的。

另外,围绕着一些有前途的新技术,总会出现大量的"快速入门手册",有些简直就是对该技术帮助文档的翻译,而且,有难度的地方没有翻译出来,大家都明白的地方翻译得非常详尽,详尽到了没有必要的地步。这种因为市场需求而产生的应景时文,大量地出现在技术文章领域。

分享到:
评论

相关推荐

    Aspect学习实例.zip

    【标题】"Aspect学习实例.zip" 提供了一个深入了解Aspect编程的实际应用案例,这主要与面向切面编程(AOP)的概念相关。面向切面编程是一种编程范式,它旨在提高软件的模块化程度,通过将关注点分离为独立的“切面”...

    Spring加Aspect的jar包

    Spring+Aspect所需jars,包括Spring-aop、Spring-Aspect、aopalliance-1.0、Aspect Jweaver-1.8.7等包,个人学习时从各地凑出来的,送上资源免得各位大大们继续跟我一样还得花时间从各地一点一点地去凑这几个包。

    springboot框架demo学习,内含filter、Interceptor、Aspect以及注解形式的aspect

    springboot框架demo,其中包含了filter、Interceptor、Aspect以及注解形式的aspect,以及对应的get和post接口,配置好了可以直接跑,省的再自己去搜索进行配置了,用来学习过滤器,拦截器和切面的配置以及执行顺序非常...

    spring aspect 配置2种方式

    在Spring框架中,Aspect是面向切面编程(AOP)的核心概念,它允许我们将关注点分离,比如日志、事务管理等,从主...在学习和使用过程中,可以通过`springaoptest1`和`springaoptest2`这两个文件进行实践操作,加深理解。

    Spring aop 记录操作日志 Aspect 源码

    在IT行业中,Spring AOP(面向切面编程)是一种强大的工具,它允许我们在代码中实现横切关注点,如日志记录、权限控制等...通过不断地学习和实践,我们可以更好地利用Spring AOP来解决实际开发中的问题,提高开发效率。

    Deep Multi-Task Learning for Aspect Term Extraction.pdf

    本文讨论了基于深度学习的多任务学习框架,特别是通过长短期记忆网络(LSTM)处理用户评论句子中的方面术语提取(Aspect Term Extraction,ATE)问题。方面术语提取是基于方面的意见挖掘(Aspect-Based Sentiment ...

    Deep_Learning_for_Aspect-Based_Sentiment_Analysis_A_Comparative_Review.pdf

    本文献的标题《深度学习用于方面级情感分析:一个比较性回顾》表明,本文献旨在比较性地回顾深度学习在方面级情感分析中的应用,作者Hai Ha Do等人来自澳大利亚悉尼的查尔斯·斯特大学。文章在2018年4月3日收到,...

    Spring 2.5 Aspect-Oriented Programming source code

    但是,这个文件很可能是 Spring 2.5 AOP 相关的源代码,可以用来深入学习和理解 Spring 如何实现 AOP 机制,以及如何在实践中应用这些概念。通过阅读和分析源代码,你可以更好地掌握 Spring AOP 的工作原理,从而在...

    java8源码-tp-practice:千里之行,始于足下

    java8 源码 tp-practice 介绍 千里之行,始于足下。 软件架构 ...aspect 学习 REST 接口多线程编程 Thread#start Thread#stop spring boot 1.5.20.RELEASE spring cloud Dalston.SR2 itplh-sso jwt + oau

    spring自定义注解+Aspect使用1

    在本文中,我们将学习如何在 Spring 项目中使用自定义注解和 Aspect 来实现日志记录功能。我们将从头开始,创建一个简单的 Spring Boot 项目,然后使用自定义注解和 Aspect 来实现日志记录。 自定义注解 首先,...

    AspectDemo

    在这个示例中,我们能看到Aspect库的三种不同实现方式,这为开发者提供了学习和参考的素材。 AOP的核心概念包括切面(Aspect)、连接点(Join Point)、通知(Advice)、切入点(Pointcut)和织入(Weaving)。 1....

    基于pytorch-transformers版本,暂时只实现了BERT做aspect+opinion

    标题中的“基于pytorch-transformers版本,暂时只实现了BERT做aspect+opinion”指的是一个使用PyTorch-Transformers库的自然语言处理项目,该项目专注于情感分析中的特定任务——方面(aspect)加观点(opinion)...

    gradle-aspect-1.6与2.1 jar 并授人以渔

    最后,使用AspectJ插件进行AOP编程时,我们需要编写Aspect类,并在Gradle构建脚本中配置AspectJ插件。例如,创建一个`MyAspect.aj`文件,然后在`build.gradle`中配置: ```groovy apply plugin: 'java' apply ...

    面向方面aspect的程序设计方法

    学习AOP可以帮助开发者更有效地组织代码,减少重复工作,并提高代码的可维护性。《面向方面aspect的程序设计方法》这本书应该会深入探讨这些概念,通过实例和实践指导读者如何在实际项目中应用AOP。对于想要提升软件...

    sgz.zip aspect 的jar 包

    在IT行业中,AOP(Aspect-Oriented Programming,面向切面编程)是一种编程范式,旨在提高代码的可重用性和可维护性,通过将关注点分离为独立的模块,称为切面。Spring框架是Java领域中最流行的轻量级开源框架之一,...

    aspectjrt1.7和aspectweaver1.

    这种方式极大地简化了切面的编写,降低了学习和使用AOP的门槛。 使用AspectJRT1.7和AspectWeaver1.0进行AOP开发,开发者可以获得以下优势: 1. **代码解耦**:将关注点分离,减少代码间的耦合度,提高代码的可读性...

    Any Aspect Ratio and Resolution.pdf

    ### Any Aspect Ratio and Resolution: NaViT - A Vision Transformer for Flexible Input Processing #### 概述 本文档介绍了NaViT(Native Resolution ViT),一种专为处理任意分辨率和宽高比图像而设计的视觉...

    毕业设计-基于主题的情感分析,LSTM,aspect embedding,添加Attention机制python源码+文档说明

    基于主题的情感分析,LSTM,aspect embedding,添加Attention机制 aspect-based sentiment analysis 是文本分类的一个子任务 输入:一段文本; 输出:主题-极性,比如:服务-积极( positive )、味道-消极( ...

    Android-logging-aspect项目中日志收集实现AOP和Listener两种全局实现

    通过学习和使用这个logging-aspect项目,开发者可以了解到如何在Android应用中实现全局的日志收集,同时理解AOP和Listener在实际开发中的应用,这对于提升Android应用的可维护性和可调试性有着积极的意义。

    aspect-based-sentiment-analysis:给定句子中的某个方面字词,预测该方面字词的情感标签

    《基于方面的情感分析:深入理解Aspect-Based Sentiment Analysis与MemNet模型》 情感分析是自然语言处理领域的一个重要分支,其目标是理解文本中的情绪色彩,而Aspect-Based Sentiment Analysis(ABSA)则是这一...

Global site tag (gtag.js) - Google Analytics