AOP in Spring
AOP in Spring
是不是已经对包裹在每个业务周围的异常处理、事务管理、性能监控、日志记录等重复出现的代码感到厌倦,那么是时候轮到AOP出场了。不得不承认程序员的惰性有时候会是一件好事(毕竟提高生产率的终极目标是增加休息时间)。有一个统计类的项目,业务过程相当复杂,从输入参数到输出结果,中间会产生大量的临时数据。客户的要求是程序需要记录下每一个中间过程的临时数据,这样方便验证统计过程是否正确。客户以前是程序员,非常迷信封闭式开发并固执地要求了解开发中的每个细节。AOP在他写代码的年代还没有,所以项目组专门为他写了一个演示程序。
定义一个Aspect类,其中包括切入点表达式和四个通知(@Before、@After、@AfterReturning、@AfterThrowing),并且该类由@Component注入到Spring容器中。
![](https://common.cnblogs.com/images/copycode.gif)
package com.mmh.aop; import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component; @Component
@Aspect
public class Aspector { private static final String pointCut = "execution(* com.mmh.business.*.*(..))"; @Before(pointCut)
public void before(JoinPoint joinPoint) {
System.out.println(joinPoint.getSignature().getName() + ": 开始执行");
} @After(pointCut)
public void after(JoinPoint joinPoint) {
System.out.println(joinPoint.getSignature().getName() + ": 结束执行");
} @AfterReturning(pointcut = pointCut, returning = "result")
public void logAfterReturning(JoinPoint joinPoint, Object result) {
System.out.println(joinPoint.getSignature().getName() + " 执行结果: "
+ result);
} @AfterThrowing(pointcut = pointCut, throwing = "error")
public void logAfterThrowing(JoinPoint joinPoint, Throwable error) {
System.out.println(joinPoint.getSignature().getName() + " 抛出异常: "
+ error);
}
}
![](https://common.cnblogs.com/images/copycode.gif)
定义一个模拟业务计算的Calculate类以及两个子业务类,Add和Divide。在实际的商业化项目中最好做这样的分解,努力保持一个类、函数和模块只完成一个任务(Single Responsibility Principle的宗旨)。这样在需求发生变化或算法出现问题时,修改代码的成本会相对较低。
![](https://common.cnblogs.com/images/copycode.gif)
package com.mmh.business; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component; @Component("Calculate")
public class Calculate { @Autowired
private Add addMethod; @Autowired
private Divide divideMethod; public int calculate(int a, int b) {
return divideMethod.divide(addMethod.add(a, b), b);
}
}
![](https://common.cnblogs.com/images/copycode.gif)
![](https://common.cnblogs.com/images/copycode.gif)
package com.mmh.business; import org.springframework.stereotype.Component; @Component("Add")
public class Add { public int add(int a, int b) {
return a + b;
}
}
![](https://common.cnblogs.com/images/copycode.gif)
![](https://common.cnblogs.com/images/copycode.gif)
package com.mmh.business; import org.springframework.stereotype.Component; @Component("Divide")
public class Divide { public int divide(int a, int b) {
return a / b;
}
}
![](https://common.cnblogs.com/images/copycode.gif)
写一个简单的测试类让程序运行起来。在程序的输出中可以清楚地看到各个子业务执行情况,这就满足了客户的需求。
![](https://common.cnblogs.com/images/copycode.gif)
package com.mmh.test; import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; import com.mmh.business.Calculate; public class Test { public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext(
"appContext.xml"); Calculate p = (Calculate) context.getBean("Calculate");
p.calculate(1, 2);
}
}
// 运行结果:
/**
* calculate: 开始执行
* add: 开始执行
* add: 结束执行
* add 执行结果: 3
* divide: 开始执行
* divide: 结束执行
* divide 执行结果: 1
* calculate: 结束执行
* calculate 执行结果: 1
*/
![](https://common.cnblogs.com/images/copycode.gif)
最后还是要把Spring配置好,让它为我们提供优秀的服务。因为在这个演示程序中使用的是注解的方式向Spring容器中注入Bean,也用注解的方式配置AOP,所以配置文件相当简单。这种简单的方式是否在任何应用场景中都适用呢?当然不是,这种方式缺乏灵活性,需求变化时,开发人员必须要修改代码以及编译代码。
![](https://common.cnblogs.com/images/copycode.gif)
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd"> <context:component-scan base-package="com.mmh" /> <aop:aspectj-autoproxy /> </beans>
![](https://common.cnblogs.com/images/copycode.gif)
AOP in Spring的更多相关文章
- 死磕Spring之AOP篇 - Spring AOP常见面试题
该系列文章是本人在学习 Spring 的过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring 源码分析 GitHub 地址 进行阅读. Spring 版本:5.1 ...
- 死磕Spring之AOP篇 - Spring AOP自动代理(一)入口
该系列文章是本人在学习 Spring 的过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring 源码分析 GitHub 地址 进行阅读. Spring 版本:5.1 ...
- 死磕Spring之AOP篇 - Spring 事务详解
该系列文章是本人在学习 Spring 的过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring 源码分析 GitHub 地址 进行阅读. Spring 版本:5.1 ...
- 【spring 5】AOP:spring中对于AOP的的实现
在前两篇博客中,介绍了AOP实现的基础:静态代理和动态代理,这篇博客介绍spring中AOP的实现. 一.采用Annotation方式 首先引入jar包:aspectjrt.jar && ...
- AOP及spring AOP的使用
介绍 AOP是一种概念(思想),并没有设定具体语言的实现. AOP是对oop的一种补充,不是取而代之. 具体思想:定义一个切面,在切面的纵向定义处理方法,处理完成之后,回到横向业务流. 特征 散布于应 ...
- SpringBoot CGLIB AOP解决Spring事务,对象调用自己方法事务失效.
对于像我这种喜欢滥用AOP的程序员,遇到坑也是习惯了,不仅仅是事务,其实只要脱离了Spring容器管理的所有对象,对于SpringAOP的注解都会失效,因为他们不是Spring容器的代理类,Sprin ...
- 浅谈spring中AOP以及spring中AOP的注解方式
AOP(Aspect Oriented Programming):AOP的专业术语是"面向切面编程" 什么是面向切面编程,我的理解就是:在不修改源代码的情况下增强功能.好了,下面在 ...
- AOP:spring 的Annotation配置
1.文件目录: 2.实体类 package com.wangcf.po; public class User { private int id; private String name; privat ...
- 【AOP】Spring AOP基础 + 实践 完整记录
Spring AOP的基础概念 ============================================================= AOP(Aspect-Oriented Pr ...
随机推荐
- 通信网Project之——单源单宿最短路问题
最主要的Vertex类: #ifndef VERTEX_H #define VERTEX_H #include <climits> #include <cstddef> #de ...
- Mediator - 中介者模式
定义 用一个中介对象来封装一系列的对象的交互.中介者使各对象不须要显示地相互使用,从而使其耦合松散,并且能够独立的改变他们之间的交互. 案例 比方有一个图像界面,在界面上有一个输入框LineEdit, ...
- ASP.NET MVC性能优化工具 MiniProfiler
ASP.NET MVC性能优化工具 MiniProfiler 2014年04月19日 ⁄ ASP.NET ⁄ 共 1159字 ⁄ 字号 小 中 大 ⁄ 暂无评论 ⁄ 阅读 325 views 次 MV ...
- cocos2d-x3.0之请求网络(phpserver)
HelloWorldScene.h #ifndef __HELLOWORLD_SCENE_H__ #define __HELLOWORLD_SCENE_H__ #include "cocos ...
- PKU 1276 Cash Machine
<span style="color:#000099;">/* Cash Machine Time Limit: 1000MS Memory Limit: 10000K ...
- iOS MapKit导航及地理转码辅助类
头文件: #import <Foundation/Foundation.h> #import <MapKit/MapKit.h> @interface DirectionRou ...
- Notepad++ 配置 支持jquery、html、css、javascript、php代码提示
原文:Notepad++ 配置 支持jquery.html.css.javascript.php代码提示 官网下载:http://notepad-plus-plus.org/ 获取插件的方法:打开软件 ...
- 微软BI 之SSIS 系列 - 使用 Script Task 访问非 Windows 验证下的 SMTP 服务器发送邮件
原文:微软BI 之SSIS 系列 - 使用 Script Task 访问非 Windows 验证下的 SMTP 服务器发送邮件 开篇介绍 大多数情况下我们的 SSIS 包都会配置在 SQL Agent ...
- Apache JMeter--网站自动测试与性能测评
Apache JMeter--网站自动测试与性能测评 2013-02-28 15:48:05 标签:Jmeter From:http://bdql.iteye.com/blog/291987 出于学习 ...
- Ibatis ISqlMapper工厂类案例
namespace Model{ public class MapperFactory { //声明一个ISqlMapper接口类型的数据映射器 _mapper,其初始值为null private s ...