一步一步深入spring(5)--使用基于注解的spring实现 AOP
1.要利用spring aop,至少需要添加以下jar包
使用spring需要的jar
spring.jar 、commons-logging.jar
使用切面编程(AOP)需要的jar
aspectjrt.jar、aspectjweaver.jar、cglib-nodep-2.1_3.jar
使用JSR-250提供的注解,如@Resource,需要的jar
common-annotations.jar
2.使用spring 进行aop编程,首先我们要在Spring的配置文件中引入aop命名空间:
<beans xmlns=http://www.springframework.org/schema/beans
xmlns:xsi=http://www.w3.org/2001/XMLSchema-instance
xmlns:aop=http://www.springframework.org/schema/aop
xsi:schemaLocation=”http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springramework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.5.xsd”> </beans>
当然我们在使用spring的时候,肯定会使用到spring ioc相关的内容,也需要加入命名空间对IOC部分的支持
3.使用注解方式来实现aop,首先需要在spring的配置文件中启动对@AspectJ注解的支持
<aop:aspectj-autoproxy/>
4.建立一个类,作为切面类,然后在切面类中依次添加切入点,前置通知,后置通知,例外通知,最终通知,环绕通知等:
完整的java文件:
package com.yangyang.aop; import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut; //声明该类为一个切面
@Aspect
public class MyInterceptor {
//切入点要拦截的类
@Pointcut("execution (* com.yangyang.service..*.*(..))")
private void anyMethod(){} //声明一个切入点,切入点的名称其实是一个方法 //前置通知(不需要获取输入参数)
@Before("anyMethod()")//第一个参数为切入点的名称
public void doAccessCheck(){
System.out.println("前置通知");
} //后置通知(不需要获取返回值)
@AfterReturning("anyMethod()")
public void doAfterReturning(){
System.out.println("后置通知:");
} //例外通知(不需要异常信息)
@AfterThrowing("anyMethod()")
public void doAfterThrowing(){
System.out.println("例外通知");
} //最终通知
@After("anyMethod()")
public void doAfter(){
System.out.println("最终通知");
} //环绕通知(特别适合做权限系统)
@Around("anyMethod()")
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable{
System.out.println("环绕通知进入方法");
Object object=pjp.proceed();
System.out.println("环绕通知退出方法");
return object;
} }
5.建立供测试的业务方法:这里只列出实现:
package com.yangyang.service.impl; import com.yangyang.service.PersonService; public class PersonServiceImpl implements PersonService{ @Override
public String getNameById(Long id) {
System.out.println("我是getNameById()方法");
return "csy";
} @Override
public void save(String name) {
throw new RuntimeException("故意抛出了异常,仅供测试");
// System.out.println("我是save()方法");
} @Override
public void update(String name, Long id) {
System.out.println("我是update()方法");
} }
6.将业务service的bean以及切面的类加入spring管理,完整的配置文件如下:
<?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-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
<!-- 加上aop的命名空间以及DTD验证 --> <aop:aspectj-autoproxy/><!-- 打开aop对@Aspectj的注解支持 ,相当于为注解提供解析功能--> <bean id="myInterceptor" class="com.yangyang.aop.MyInterceptor"></bean> <bean id="personService" class="com.yangyang.service.impl.PersonServiceImpl">
</bean> </beans>
7.编写单元测试,来测试aop是否成功实现。代码:
@Test
public void testSpringInterceptor() {
ApplicationContext ctx=new ClassPathXmlApplicationContext("resources/beans.xml");
PersonService personService=(PersonService)ctx.getBean("personService");
personService.getNameById(1l);
System.out.println("----------------------------------");
personService.save("csy");
}
观察结果如下:
前置通知
环绕通知进入方法
我是getNameById()方法
后置通知:
最终通知
环绕通知退出方法
----------------------------------
前置通知
环绕通知进入方法
例外通知
最终通知
可以看出AOP编写成功了。
思考一个问题,如果我们希望获取在业务操作在使用aop的服务时得到相应的输入,返回值,以及异常信息,那么切面该怎么改呢:
只需要在各自通知的地方加入相应类型的参数即可:
package com.yangyang.aop; import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut; //声明该类为一个切面
@Aspect
public class MyInterceptor {
//切入点要拦截的类
@Pointcut("execution (* com.yangyang.service..*.*(..))")
private void anyMethod(){} //声明一个切入点,切入点的名称其实是一个方法 //前置通知(获取输入参数)
@Before("anyMethod() && args(name)")//第一个参数为切入点的名称,第二个是测试获取输入参数,此处为string类型的,参数名称与方法中的名称相同,如果不获取输入参数,可以不要
public void doAccessCheck(String name){
System.out.println("前置通知:"+name);
} //后置通知(获取返回值)
@AfterReturning (pointcut="anyMethod()", returning="result")
public void doAfterReturning(String result){
System.out.println("后置通知:"+result);
} //例外通知(获取异常信息)
@AfterThrowing(pointcut="anyMethod()",throwing="e")
public void doAfterThrowing(Exception e){
System.out.println("例外通知:"+e);
} //最终通知
@After("anyMethod()")
public void doAfter(){
System.out.println("最终通知");
} //环绕通知(特别适合做权限系统)
@Around("anyMethod()")
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable{
System.out.println("环绕通知进入方法");
Object object=pjp.proceed();
System.out.println("环绕通知退出方法");
return object;
} }
同理执行单元测试,可以看到结果:
环绕通知进入方法
我是getNameById()方法
后置通知:csy
最终通知
环绕通知退出方法
----------------------------------
前置通知:csy
环绕通知进入方法
例外通知:java.lang.RuntimeException: 故意抛出了异常,仅供测试
最终通知
这样我们的功能就开发完了
一步一步深入spring(5)--使用基于注解的spring实现 AOP的更多相关文章
- 基于注解的Spring AOP的配置和使用
摘要: 基于注解的Spring AOP的配置和使用 AOP是OOP的延续,是Aspect Oriented Programming的缩写,意思是面向切面编程.可以通过预编译方式和运行期动态代理实现在不 ...
- Spring7:基于注解的Spring MVC(下篇)
Model 上一篇文章<Spring6:基于注解的Spring MVC(上篇)>,讲了Spring MVC环境搭建.@RequestMapping以及参数绑定,这是Spring MVC中最 ...
- Spring6:基于注解的Spring MVC(上篇)
什么是Spring MVC Spring MVC框架是一个MVC框架,通过实现Model-View-Controller模式来很好地将数据.业务与展现进行分离.从这样一个角度来说,Spring MVC ...
- 基于注解的Spring AOP示例
基于注解的Spring AOP示例 目录 在XML配置文件中开启 @AspectJ 支持 声明切面及切入点 声明通知 测试 结语 在XML配置文件中开启 @AspectJ 支持 要使用Spring的A ...
- Spring MVC中基于注解的 Controller
终于来到了基于注解的 Spring MVC 了.之前我们所讲到的 handler,需要根据 url 并通过 HandlerMapping 来映射出相应的 handler 并调用相应的方法以响 ...
- 使用 Spring 2.5 基于注解驱动的 Spring MVC
http://www.ibm.com/developerworks/cn/java/j-lo-spring25-mvc/ 概述 继 Spring 2.0 对 Spring MVC 进行重大升级后,Sp ...
- 使用 Spring 2.5 基于注解驱动的 Spring MVC--转
概述 继 Spring 2.0 对 Spring MVC 进行重大升级后,Spring 2.5 又为 Spring MVC 引入了注解驱动功能.现在你无须让 Controller 继承任何接口,无需在 ...
- 基于注解的Spring多数据源配置和使用(非事务)
原文:基于注解的Spring多数据源配置和使用 1.创建DynamicDataSource类,继承AbstractRoutingDataSource package com.rps.dataSourc ...
- Spring:基于注解的Spring MVC
什么是Spring MVC Spring MVC框架是一个MVC框架,通过实现Model-View-Controller模式来很好地将数据.业务与展现进行分离.从这样一个角度来说,Spring MVC ...
随机推荐
- DirectX11 学习笔记2 - 加入关键事件 实现视角转换 旋转
上的程序的的基础上.在基类D3DBase添加摄像头功能 //录影机 void D3DBase::setCamera() { //关键事件 //假定A,S,D,W,Q,E,Z,X,C键被按下.动摄像机 ...
- Codeforces 474A Keyboard (水
题目链接:点击打开链接 键盘移位了,问输出相应的字母 #include <cstdio> #include <cstring> char a[105]; char b[3][1 ...
- SQL Server 2012的附件失败,与硬链接的问题
1.我在系统上做了硬链接,也就是把C盘的某个目录,映射为D盘. 2.把数据库文件mdf和ldf放入D盘.结果,SQL Server的企业管理器,无法从D盘里加载mdf或ldf文件,并且在目录下无法显示 ...
- PhpStorm创建Drupal模块项目开发教程
在PhpStorm开发工具中,创建Drupal开发项目有两种方式:整合Drupal到现有的项目中和直接创建一个新的Drupal模块. 接下来将展示这两种方式的具体操作! 整合Drupal到现有的项目 ...
- 5款新颖的ReSharper插件
ReSharper是著名的代码生成工具.自ReSharper 8.0版本发布以来,新的扩展管理器.模板设置.分发安装等功能改革使得对插件的要求变高.接下来小编整理了5款新颖插件. JetBox 当开发 ...
- AJAX入门——工作原理
同步和异步交互,了解互动 对于一个样本:一般B/S模式(同步) AJAX技术(异步) * 同步: 提交请求->等待server处理->处理完成返回 ...
- Centos中如何配置Texlive2013中文字体的问题
Centos中如何配置Texlive2013中文字体的问题: 第一步是下载你需要的字体,我从windows/fonts中拷贝的比较多,你只要复制你需要的字体即可. 注意只要文件扩展名为ttf的文件,t ...
- 【C#版本详情回顾】C#3.0主要功能列表
隐式类型的本地变量和数组 在与本地变量一起使用时,var 关键字指示编译器根据初始化语句右侧的表达式推断变量或数组元素的类型 对象初始值设定项 支持无需显式调用构造函数即可进行对象初始化 集合初始值设 ...
- INSTEAD OF触发器
Oracle触发器5(INSTEAD OF触发器) 前提:对于简单的视图,可以直接进行DML操作,但是对于复杂视图,不允许直接执行DML操作,当视图符合以下任何一种情况都不可以: 具有集合操作符(UN ...
- 订单处理(c#实现)
State模式的经典应用场景:订单处理(c#实现) State模式在对象内部状态发生变化的时候,改变自身的行为,这通常是通过切换内部状态对象实现的,对象将自身在各个状态的行为推给了状态对象,从而解开了 ...