文档章节

Spring AOP 的实现方式(以日志管理为例)

Jeam_
 Jeam_
发布于 2018/07/04 11:16
字数 2178
阅读 123
收藏 10

Spring AOP 的实现方式(以日志管理为例)

2016年10月08日 00:13:57

阅读数:23198

在学习Spring框架的历程中,最重要的是要理解Spring的IOC和AOP了,不但要学会怎么用,最好是知道它是怎么实现的,通过这个国庆假期,好好地过了一下spring的AOP的皮毛,故记录一下学习心得。

一、为什么需要AOP

假如我们应用中有n个业务逻辑组件,每个业务逻辑组件又有m个方法,那现在我们的应用就一共包含了n*m个方法,我会抱怨方法太多。。。现在,我有这样一个需求,每个方法都增加一个通用的功能,常见的如:事务处理,日志,权限控制。。。最容易想到的方法,先定义一个额外的方法,实现该功能,然后再每个需要实现这个功能的地方去调用这个额外的方法。这种做法的好处和坏处分别是。
好处:可以动态地添加和删除在切面上的逻辑而不影响原来的执行代码。
坏处:一旦要修改,就要打开所有调用到的地方去修改。
好,现在我们用AOP的方式可以实现在不修改源方法代码的前提下,可以统一为原多个方法增加横切性质的“通用处理”。

二、什么是AOP

都说AOP好用,那现在我们来谈谈什么是AOP。

AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。本文提供Spring官方文档出处:Aspect Oriented Programming with Spring

从官方文档上摘抄的解释就是:面向方面编程(AOP)是面向对象编程(OOP)补充的另一种提供思考程序结构补充。在OOP中模块化的关键单元是类,而在AOP模块的单位是一个方面。面对关注点,如事务管理跨越多个类型和对象切模块化。(这些关注经常被称为在AOP文学横切关注点。)

相关概念(只需做个大概的了解就好)----来自于官方文档直译(本人英文水平有限。。。):

Aspect:这横切多个对象关心的模块化。事务管理是企业Java应用程序的横切关注点的一个很好的例子。在Spring AOP中,切面可以使用类(基于模式)或@Aspect注解(@AspectJ风格)注解普通班实施。
Join point:程序在执行过程中的一个点,如方法的执行或异常的处理。在Spring AOP中,一个连接点总是代表一个方法的执行。
Advice:在切面的某个特定的动作连接点。不同类型的意见,包括 "around," "before" and "after"的advice。 (通知的类型将在下面讨论)。许多AOP框架,包括Spring都是以拦截器作为通知模型,去维护一条围绕着一个连接点的拦截器链。
Pointcut:匹配连接点的断言。通知是跟一个切入点表达式,并在运行在切入点匹配的连接点相关联(例如,一个方法的执行要有一个确定的名字)。切入点表达式作为匹配的连接点的概念是重要的对AOP和Spring缺省使用AspectJ切入点表达式语言。
Introduction:声明代表的类型的额外的方法或字段。 Spring允许引入新的接口(以及一个对应的实现)到任何被代理的对象。例如,你可以使用引入来使一个bean实现IsModified接口,以便简化缓存。 (介绍被誉为AspectJ的社会类型间的声明。)
Target object:对象由一个或多个方面被建议。也被称作被通知对象。既然Spring AOP是通过运行时代理实现的,这个对象永远是一个被代理对象。
AOP proxy:AOP框架,以实现切面契约(例如通知方法执行等等)创建的对象。在Spring中,AOP代理可以是JDK动态代理或者CGLIB代理。
Weaving:与连接其他应用程序类型或对象方面来创建一个被通知的对象。这是可以做到在编译时(使用AspectJ编译器,例如),加载时间,或在运行时。 Spring AOP中,像其他纯Java AOP框架,在运行时进行编织。

以上概念个人感觉在做实验的时候就差不多理解了,不需要咬文嚼字地区啃。

那问题来了,AOP是在什么时候去改我们的代码的?即给我们加上额外的横切性质的"通用处理"的?

两个时机:

1.在编译java源代码的时候 ----编译时增强
2.在运行时动态地修改类 ----运行时增强(动态代理)

我们的Spring的AOP的实现原理就是基于动态代理。(之后我会尝试一下跟随源码看看Spring在AOP方面做了哪些事情)

三、Spring AOP的3种实现方式

对于框架的学习,我觉得得先会用,然后再深入原理。关于Spring AOP的实现我在这里划分成3个方式(以日志管理为例)废话不多说,直接上代码了。(以下代码是基于我之前所写的SSM框架整合的例子,如果有需要可查看我之前的博客)

配置之前注意配置文件要加上命名空间:xmlns:aop="http://www.springframework.org/schema/aop"

1.基于xml配置的实现

spring-mvc.xml

 

 
  1. <!-- 使用xml配置aop -->

  2. <!-- 强制使用cglib代理,如果不设置,将默认使用jdk的代理,但是jdk的代理是基于接口的 -->

  3. <aop:config proxy-target-class="true" />

  4. <aop:config>

  5. <!--定义切面-->

  6. <aop:aspect id="logAspect" ref="logInterceptor">

  7. <!-- 定义切入点 (配置在com.gray.user.controller下所有的类在调用之前都会被拦截)-->

  8. <aop:pointcut expression="execution(* com.gray.user.controller.*.*(..))" id="logPointCut"/>

  9. <!--方法执行之前被调用执行的-->

  10. <aop:before method="before" pointcut-ref="logPointCut"/><!--一个切入点的引用-->

  11. <aop:after method="after" pointcut-ref="logPointCut"/><!--一个切入点的引用-->

  12. </aop:aspect>

  13. </aop:config>

LogInterceptor.java

 

 

 
  1. package com.gray.interceptor;

  2.  
  3. import org.springframework.stereotype.Component;

  4. import org.slf4j.Logger;

  5. import org.slf4j.LoggerFactory;

  6.  
  7. @Component

  8. public class LogInterceptor {

  9. private final Logger logger = LoggerFactory.getLogger(LogInterceptor.class);

  10. public void before(){

  11. logger.info("login start!");

  12. }

  13.  
  14. public void after(){

  15. logger.info("login end!");

  16. }

  17. }

在这里我没有配bean是因为我在之前的配置文件里面写了自动扫描组件的配置了
要加入日志管理逻辑的地方

 

 

 
  1. @RequestMapping("/dologin.do") //url

  2. public String dologin(User user, Model model){

  3. logger.info("login ....");

  4. String info = loginUser(user);

  5. if (!"SUCC".equals(info)) {

  6. model.addAttribute("failMsg", "用户不存在或密码错误!");

  7. return "/jsp/fail";

  8. }else{

  9. model.addAttribute("successMsg", "登陆成功!");//返回到页面说夹带的参数

  10. model.addAttribute("name", user.getUsername());

  11. return "/jsp/success";//返回的页面

  12. }

  13. }

 

结果截图:

2.基于注解的实现

spring-mvc.xml

 

 
  1. <aop:aspectj-autoproxy proxy-target-class="true">

  2. </aop:aspectj-autoproxy>

 

LogInterceptor.java

 

 
  1. @Aspect

  2. @Component

  3. public class LogInterceptor {

  4. private final Logger logger = LoggerFactory.getLogger(LogInterceptor.class);

  5. @Before(value = "execution(* com.gray.user.controller.*.*(..))")

  6. public void before(){

  7. logger.info("login start!");

  8. }

  9. @After(value = "execution(* com.gray.user.controller.*.*(..))")

  10. public void after(){

  11. logger.info("login end!");

  12. }

  13. }

要加入逻辑的地方同上。

结果截图:

3.基于自定义注解的实现

基于注解,所以spring-mvc.xml也是和上面的一样的。

LogInterceptor.java(这里我只加入前置日志)

 

 
  1. package com.gray.interceptor;

  2.  
  3. import java.lang.reflect.Method;

  4.  
  5. import org.aspectj.lang.JoinPoint;

  6. import org.aspectj.lang.annotation.Aspect;

  7. import org.aspectj.lang.annotation.Before;

  8. import org.aspectj.lang.annotation.Pointcut;

  9. import org.aspectj.lang.reflect.MethodSignature;

  10. import org.slf4j.Logger;

  11. import org.slf4j.LoggerFactory;

  12. import org.springframework.stereotype.Component;

  13.  
  14. import com.gray.annotation.Log;

  15.  
  16. @Aspect

  17. @Component

  18. public class LogInterceptor {

  19. private final Logger logger = LoggerFactory.getLogger(LogInterceptor.class);

  20.  
  21. @Pointcut("@annotation(com.gray.annotation.Log)")

  22. public void controllerAspect() {

  23.  
  24. }

  25. @Before("controllerAspect()")

  26. public void before(JoinPoint joinPoint){

  27. logger.info(getOper(joinPoint));

  28. }

  29. private String getOper(JoinPoint joinPoint) {

  30. MethodSignature methodName = (MethodSignature)joinPoint.getSignature();

  31. Method method = methodName.getMethod();

  32. return method.getAnnotation(Log.class).oper();

  33. }

  34. }

同时,加入逻辑的地方需要加入Log注解

 
  1. @RequestMapping("/dologin.do") //url

  2. @Log(oper="user login")

  3. public String dologin(User user, Model model){

  4. logger.info("login ....");

  5. String info = loginUser(user);

  6. if (!"SUCC".equals(info)) {

  7. model.addAttribute("failMsg", "用户不存在或密码错误!");

  8. return "/jsp/fail";

  9. }else{

  10. model.addAttribute("successMsg", "登陆成功!");//返回到页面说夹带的参数

  11. model.addAttribute("name", user.getUsername());

  12. return "/jsp/success";//返回的页面

  13. }

  14. }

 

结果截图:

以上就是我总结的SpringAOP的3种实现方式,对于这我还有点话要说,常见的基于注解的方式除了before和after,还有around和AfterThrowing等,在做第三种方式实验的时候还遇到这种错误:error at ::0 can't find referenced pointcut。

我的解决办法是:因为我的jdk是1.7,与原来在pom.xml中配的那个aspectjweaver的jar包版本不匹配,所以我需要升级,由1.5.4改成1.7.4。

此外多说一句,对于日志,权限等业务逻辑很多人其实也喜欢用拦截器来实现的。

本文转载自:https://blog.csdn.net/donggua3694857/article/details/52752503

共有 人打赏支持
Jeam_
粉丝 1
博文 32
码字总数 24208
作品 0
长沙
程序员
私信 提问
Java设计模式综合运用(动态代理+Spring AOP)

AOP设计模式通常运用在日志,校验等业务场景,本文将简单介绍基于Spring的AOP代理模式的运用。 1. 代理模式 1.1 概念 代理(Proxy)是一种提供了对目标对象另外的访问方式,即通过代理对象访问...

landy8530
2018/10/05
0
0
Java程序员从笨鸟到菜鸟之(六十七)细谈Spring(一)spring简介

Spring 是一个开源框架,是为了解决企业应用程序开发复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许您选择使用哪一个组件,同时为 J2EE 应用程序开发提供集成的框架。 然而...

长平狐
2012/11/12
72
0
Spring AOP进行日志记录,管理

在java开发中日志的管理有很多种。我一般会使用过滤器,或者是Spring的拦截器进行日志的处理。如果是用过滤器比较简单,只要对所有的.do提交进行拦截,然后获取action的提交路径就可以获取对...

长平狐
2013/01/06
26.7K
1
spring 学习(三):aop 学习

spring 学习(三):aop 学习 aop 概念 1 aop:面向切面(方面)编程,扩展功能不修改源代码实现 2 AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码 3 aop底层使用动态代理实现 (1...

希希里之海
2018/08/19
0
0
简单理解AOP(面向切面编程)

AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。   AOP与OOP是面向不同领域的两种设计思想。   OOP(...

lisn
2015/07/01
0
0

没有更多内容

加载失败,请刷新页面

加载更多

什么是JSONP?

json底层原理: 1.他是利用<script/>里的“src”标签能 进行跨域请求的特性加载资源,但是他加载到的资源会当作一个js脚本解析,所以我们得给他返回一个js脚本。 2.所以我们用一个方法名“f...

红土豆
24分钟前
2
0
ByteBuffer详解

在Java nio中,主要有三大组件:Buffer,Channel和Selector。这三者之间的关系可以按照如下方式进行理解: Buffer提供了一个字节缓冲区,其可以不断的从Channel中读取接收到的数据。Buffer的...

爱宝贝丶
24分钟前
4
0
Maven【私有仓库、上传jar包、引用私服jar包、上传本地项目到私服】

搭建私有服务器 前面已经说过了,我们使用Maven的使用,如果需要导入相对应的jar包,Maven首先会在我们的本地仓库中寻找—>私有仓库—>中心仓库… 然而,我们的本地仓库常常没有想要的jar包的...

Nonry
40分钟前
3
0
VARCHART XGantt实践:兼顾清晰和细节的排列优化

VARCHART XGantt是一款功能强大的甘特图控件,其模块化的设计让您可以创建满足需要的应用程序。XGantt可用于.NET,ActiveX和ASP.NET应用程序,可以快速、简单地集成到您的应用程序中,帮助您...

ymy_666666
41分钟前
1
0
Syncfusion教程:在Xamarin.Forms中创建数据输入表单 (1)

下载Essential Studio for Xamarin最新版本 Essential Studio for Xamarin是全面的Xamarin.iOS、Xamarin.Android和Xamarin.Forms组件套包,包含最快的图表和网格。 介绍 欢迎学习使用Syncfus...

电池盒
41分钟前
2
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部