spring切面编程AOP 范例一
参照网上的spring AOP编程实例进行配置,但是碰到了几个坑。这篇文章重点讲解一下我踩过的两个坑:
1.使用@Service自动装配的时候,基础扫描包配置要正确;
2.xml中切面配置中的execution表达式的匹配规则要注意;
针对第一个问题
是我发现我新建的一个包里面写了一个接口和这个接口的实现,在接口实现上使用@Service,那么在使用这个service的时候,bean会自动注入进来的,但是我写的单测用例一直提示我bean注入失败。我写的其他包里的service就可以正常使用。我突然想起来,spring要识别@Service,是需要扫描基础包的,那应该是我的基础包配置没有将我新建的这个包包含进去。
applicationContext.xml中最初的配置:
<context:component-scan base-package="com.nuomi"/>
而我新建的包名是:com.baidu.nuomi.crm.mm,果然是不包含在基础包里面的,于是我在applicationContext.xml配置文件中又新加了下面这一行:
<context:component-scan base-package="com.baidu"/>
就可以了。第一个问题完美解决。
下面说下第二个问题
第一步:写好切面类:
package com.baidu.nuomi.crm.helper.monitorproxy.aspect; import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint; /**
* @author sonofelice
* @version 2015年12月17日 下午1:21:46 类说明 :
*/
public class ProxyAspect {
public void doAfter(JoinPoint jp) {
System.out.println("log Ending method:" + jp.getTarget().getClass().getName() + "."
+ jp.getSignature().getName());
} public Object doAround(ProceedingJoinPoint pjp) throws Throwable {
long time = System.currentTimeMillis();
Object retVal = pjp.proceed();
time = System.currentTimeMillis() - time;
System.out.println("process time:" + time + "ms");
return retVal;
} public void doBefore(JoinPoint jp) {
System.out.println("long Beginning methoe:" + jp.getTarget().getClass().getName() + "."
+ jp.getSignature().getName());
} public void doThrowing(JoinPoint jp, Throwable ex) {
System.out.println("methoe" + jp.getTarget().getClass().getName() + "." + jp.getSignature().getName()
+ " throw exception");
System.out.println(ex.getMessage());
}
}
第二步:写好切面要织入的目标对像:
接口和其对应的实现:
package com.baidu.nuomi.crm.target.asp; /**
* @author sonofelice
* @version 2015年12月17日 下午2:49:20 类说明 :
*/
public interface AService { public void barA(); public void fooA(String _msg);
} package com.baidu.nuomi.crm.target.asp; import org.springframework.stereotype.Service; /**
* @author sonofelice
* @version 2015年12月17日 下午3:41:05 类说明 :
*/
@Service("aService")
public class AServiceImp implements AService { @Override
public void barA() {
System.out.println("AServiceImpl.barA()");
} @Override
public void fooA(String _msg) {
System.out.println("AServiceImpl.fooA(msg:" + _msg + ")");
} }
第三步:xml文件配置:
首先在spring相关的配置文件中新建一个spring-aspect.xml文件:
<?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: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/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
">
<bean id="proxyAspect" class="com.baidu.nuomi.crm.helper.monitorproxy.aspect.ProxyAspect"/>
<aop:config>
<aop:aspect id="doOpAspect" ref="proxyAspect">
<aop:pointcut id="entrance" expression="execution(* com.baidu.nuomi.crm.target.asp.*.*(..))" />
<aop:before pointcut-ref="entrance" method="doBefore"/>
<aop:after pointcut-ref="entrance" method="doAfter"/>
<aop:around pointcut-ref="entrance" method="doAround"/>
<aop:after-throwing pointcut-ref="entrance" method="doThrowing" throwing="ex"/>
</aop:aspect>
</aop:config>
</beans>
然后在applicationContext.xml文件中引入上面的配置文件:
<import resource="spring-monitorproxy.xml" />
第四步:写个单测用例看一下配置是否成功了(我用的是junit)
下面是我的单测用例:
public class BServiceTest extends BaseTest {
@Autowired
AService aService;
@Test
public void test() {
aService.barA();
aService.fooA("msg");
}
}
运行结果如下:
long Beginning methoe:com.baidu.nuomi.crm.target.asp.AServiceImp.barA
AServiceImpl.barA()
process time:0ms
log Ending method:com.baidu.nuomi.crm.target.asp.AServiceImp.barA
long Beginning methoe:com.baidu.nuomi.crm.target.asp.AServiceImp.fooA
AServiceImpl.fooA(msg:msg)
process time:0ms
log Ending method:com.baidu.nuomi.crm.target.asp.AServiceImp.fooA
当然,一开始肯定不是这么顺利的出结果的。切面类中的方法中打印的东西在我的控制台中一条都没有打印,这是为什么呢?肯定是切面类没有织入到目标类里面啊。仔细检查下配置文件,跟借鉴的别人的配置是一样的,配置第一行,都是先配置切面类的bean,不同的地方就是这个execution表达式
<aop:pointcut id="entrance" expression="execution(* com.baidu.nuomi.crm.target.asp.*.*(..))" />
那应该是我的表达式不正确。参考文章:http://lavasoft.blog.51cto.com/62575/172292/
任意公共方法的执行:
execution(public * *(..))
任何一个名字以“set”开始的方法的执行:
execution(* set*(..))
AccountService接口定义的任意方法的执行:
execution(* com.xyz.service.AccountService.*(..))
在service包中定义的任意方法的执行:
execution(* com.xyz.service.*.*(..))
在service包或其子包中定义的任意方法的执行:
execution(* com.xyz.service..*.*(..))
我想要在包asp中的定义的任意方法上执行切面类,就按照对应的规则去配置才可以成功的。
新手可以参考上面的配置,有问题可以留言。
spring切面编程AOP 范例一的更多相关文章
- Spring切面编程AOP
- Spring学习手札(二)面向切面编程AOP
AOP理解 Aspect Oriented Program面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术. 但是,这种说法有些片面,因为在软件工程中,AOP的价值体现的并 ...
- Spring学习笔记:面向切面编程AOP(Aspect Oriented Programming)
一.面向切面编程AOP 目标:让我们可以“专心做事”,避免繁杂重复的功能编码 原理:将复杂的需求分解出不同方面,将公共功能集中解决 *****所谓面向切面编程,是一种通过预编译方式和运行期动态代理实现 ...
- spring框架(2)— 面相切面编程AOP
spring框架(2)— 面相切面编程AOP AOP(Aspect Oriented Programming),即面向切面编程. 可以说是OOP(Object Oriented Programming ...
- Spring框架学习笔记(2)——面向切面编程AOP
介绍 概念 面向切面编程AOP与面向对象编程OOP有所不同,AOP不是对OOP的替换,而是对OOP的一种补充,AOP增强了OOP. 假设我们有几个业务代码,都调用了某个方法,按照OOP的思想,我们就会 ...
- Spring之控制反转——IoC、面向切面编程——AOP
控制反转——IoC 提出IoC的目的 为了解决对象之间的耦合度过高的问题,提出了IoC理论,用来实现对象之间的解耦. 什么是IoC IoC是Inversion of Control的缩写,译为控制 ...
- 04 Spring:01.Spring框架简介&&02.程序间耦合&&03.Spring的 IOC 和 DI&&08.面向切面编程 AOP&&10.Spring中事务控制
spring共四天 第一天:spring框架的概述以及spring中基于XML的IOC配置 第二天:spring中基于注解的IOC和ioc的案例 第三天:spring中的aop和基于XML以及注解的A ...
- Spring框架系列(4) - 深入浅出Spring核心之面向切面编程(AOP)
在Spring基础 - Spring简单例子引入Spring的核心中向你展示了AOP的基础含义,同时以此发散了一些AOP相关知识点; 本节将在此基础上进一步解读AOP的含义以及AOP的使用方式.@pd ...
- [译]如何在ASP.NET Core中实现面向切面编程(AOP)
原文地址:ASPECT ORIENTED PROGRAMMING USING PROXIES IN ASP.NET CORE 原文作者:ZANID HAYTAM 译文地址:如何在ASP.NET Cor ...
随机推荐
- android延迟执行
延迟执行可以通过以下操作实现,按照推荐的顺序列出来 1. new Handler().postDelayed(new Runnable(){ public void run() { //execut ...
- SQL Server事务、隔离级别详解(二十九)
前言 事务一直以来是我最薄弱的环节,也是我打算重新学习SQL Server的出发点,关于SQL Server中事务将分为几节来进行阐述,Always to review the basics. 事务简 ...
- 在C++中反射调用.NET(二)
反射调用返回复杂对象的.NET方法 定义数据接口 上一篇在C++中反射调用.NET(一)中,我们简单的介绍了如何使用C++/CLI并且初步使用了反射调用.NET程序集的简单方法,今天我们看看如何在C+ ...
- SQL 增删改查45道题
create database School use School go create table Student --1.学生表 ( Sno ) not null primary key,--学号( ...
- php下安装动态扩展库的相关事项
php下安装动态扩展库的相关事项 我下载的Apache版本为2.4,PHP版本为7.0. 将Apache与PHP集成配置好后(PHP安装目录为:G:\computer\web\php7,apache安 ...
- GitLab Wiki 内容恢复版本管理
原来一直在网站上写Wiki文档, 最近手欠误删一篇文档, 想要恢复文档时才发现原来gitlab的Wiki是用git管理的从此再也不用为误删担心了 实现步骤: mac系统安装gollow brew in ...
- 【小梅哥FPGA进阶教程】第九章 基于串口猎人软件的串口示波器
九.基于串口猎人软件的串口示波器 1.实验介绍 本实验,为芯航线开发板的综合实验,该实验利用芯航线开发板上的ADC.独立按键.UART等外设,搭建了一个具备丰富功能的数据采集卡,芯航线开发板负责进行数 ...
- 基于canvas和jsp的头像剪辑上传
最近在做项目时候需要一个头像长传功能,但是现在照片动不动就几兆的,都是流量的浪费. 我只是简单想要上传一个头像而已... 经过几天发愤图强..总算是略有所获.. 基本思路: 1.html部分,图片剪辑 ...
- php基础知识掌握——四种界定符
闲来无事,研究了与java.c#完全不同的领域php,php即(外文名:PHP: Hypertext Preprocessor,中文名:"超文本预处理器"),对于学习php的初学者 ...
- python报错UnicodeDecodeError: 'ascii' codec can't decode byte 0xe8 in position 0 解决方案
环境:mac+python 2.7 场景描述:在使用python修改excel内容修改表格内容为中文保存时报以下错误 此时已经设置了utf-8了 但保存时仍然报错错 此时将python中的中文使用un ...