Spring aop--几种不同的使用方式

标签: spring aop | 发表时间:2015-06-01 01:48 | 作者:jly4758
出处:http://blog.csdn.net

        最近听老师讲关于AOP关于容器的课,突然觉得之前对aop的理解都不是很到位,先不说理解就是应用都很少,最近也写了几篇关于AOP的博客,都是一些简单的demo,今天在这里再说一下关于Spring AOP的几种实现方式。

        我们经常会用到的有如下几种

        1、基于代理的AOP

        2、纯简单java对象切面

        3、@Aspect注解形式的

        4、注入形式的Aspcet切面

下面我们就一个一个来应用吧.

下面先写一下几个基本的类。

接口类:

/**
 * 定义一个接口
 * @author 陈丽娜
 * @version 2015年5月31日上午9:16:50
 */
public interface Sleepable {

	/**
	 * 睡觉方法
	 * @author 陈丽娜
	 * @version 2015年5月31日上午9:17:14
	 */
	void sleep();
}

实现类:

/**
 * 陈丽娜 本人实现睡觉接口
 * @author 陈丽娜
 * @version 2015年5月31日下午4:51:43
 */
public class ChenLliNa implements Sleepable {

	@Override
	public void sleep() {
		// TODO Auto-generated method stub
        System.out.println("乖,该睡觉了!");
	}
}


增强类:

/**
 * 定义一个睡眠的增强 同时实现前置 和后置
 * @author 陈丽娜
 * @version 2015年5月31日上午9:24:43
 */
public class SleepHelper implements MethodBeforeAdvice, AfterReturningAdvice {

	@Override
	public void afterReturning(Object returnValue, Method method,
			Object[] args, Object target) throws Throwable {
		 System.out.println("睡觉前要敷面膜");
	}

	@Override
	public void before(Method method, Object[] args, Object target)
			throws Throwable {
		System.out.println("睡觉后要做美梦");
	}

}

一、基于代理的AOP

 

  

   <!-- 创建一个增强 advice -->
       <bean id ="sleepHelper" class="com.tgb.springaop.aspect.SleepHelper"/>

       <bean id="lina" class="com.tgb.springaop.service.impl.ChenLliNa"/>
       <!-- 定义切点   匹配所有的sleep方法-->
       <bean id ="sleepPointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut">
              <property name="pattern" value=".*sleep"></property>
       </bean>
       
       <!-- 切面    增强+切点结合 -->
       <bean id="sleepHelperAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor">
            <property name="advice" ref="sleepHelper"/>
            <property name="pointcut" ref="sleepPointcut"/>
       </bean>
       
       <!-- 定义代理对象 -->
       <bean id="linaProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
               <property name="target" ref="lina"/>
               <property name="interceptorNames" value="sleepHelperAdvisor"/>
               <!-- <property name="proxyInterfaces" value="com.tgb.springaop.service.Sleepable"/> -->
       </bean>

如配置文件中:

pattern属性指定了正则表达式,他匹配所有的sleep方法

使用org.springframework.aop.support.DefaultPointcutAdvisor的目的是为了使切点和增强结合起来形成一个完整的切面

最后配置完后通过org.springframework.aop.framework.ProxyFactoryBean产生一个最终的代理对象。

 

二、纯简单java对象切面

纯简单java对象切面这话怎么说呢,在我看来就是相对于第一种配置,不需要使用代理,,而是通过spring的内部机制去自动扫描,这时候我们的配置文件就该如下修改:

 

       <!-- 创建一个增强 advice -->
       <bean id ="sleepHelper" class="com.tgb.springaop.aspect.SleepHelper"/>
       <!-- 目标类 -->
       <bean id="lina" class="com.tgb.springaop.service.impl.ChenLliNa"/>
      
       <!--  配置切点和通知-->
       <bean id ="sleepAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
            <property name="advice" ref="sleepHelper"></property>
            <property name="pattern" value=".*sleep"/>
       </bean>
       
       <!-- 自动代理配置 -->
       <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"/>

是不是相对于第一种简单了许多,不用再去配置代理了。

 

三、@Aspect注解形式

根据我们的经验也知道,注解的形式相对于配置文件是简单一些的,这时候需要在已有的方法或类上家注解:

/**
 * 通过注解的方式 添加增强
 * @author 陈丽娜
 * @version 2015年5月31日上午10:26:13
 */
@Aspect
@Component
public class SleepHelper03 {	
	
	/*@Pointcut("execution(* com.tgb.springaop.service.impl..*(..))")*/
	@Pointcut("execution(* *.sleep(..))")
	public void sleeppoint(){}
	
	@Before("sleeppoint()")
	public void beforeSleep(){
		System.out.println("睡觉前要敷面膜");
	}
	
	@AfterReturning("sleeppoint()")
	public void afterSleep(){
		System.out.println("睡觉后要做美梦");
	}


配置文件中只需写:

  <!--扫描包 -->
       <context:component-scan base-package="com.tgb" annotation-config="true"/> 
       <!-- ASPECTJ注解 -->
       <aop:aspectj-autoproxy  proxy-target-class="true" />  
       
       <!-- 目标类 -->
        <bean id="lina" class="com.tgb.springaop.service.impl.ChenLliNa"/> 


 

四、注入形式的Aspcet切面

个人感觉这个是最简单的也是最常用的,也是最灵活的。配置文件如下:

   <!-- 目标类 -->
       <bean id="lina" class="com.tgb.springaop.service.impl.ChenLliNa"/>
       <bean id ="sleepHelper" class="com.tgb.springaop.aspect.SleepHelper02"/>
       
       <aop:config>
           <aop:aspect ref="sleepHelper">
                <aop:before method="beforeSleep" pointcut="execution(* *.sleep(..))"/>
                <aop:after method="afterSleep" pointcut="execution(* *.sleep(..))"/>
           </aop:aspect>
       </aop:config>


配置文件中提到的SleepHelper02类如下:

/**
 * 通过注解的方式 添加增强
 * @author 陈丽娜
 * @version 2015年5月31日上午10:26:13
 */

public class SleepHelper02 {
	public void beforeSleep(){
		System.out.println("睡觉前要敷面膜");
	}
	public void afterSleep(){
		System.out.println("睡觉后要做美梦");
	}
}

 

是不是看上去都很简单呀,这样是不是大家都会使用spring aop了?!

 

关于如何调用,这里写了几个测试类,可以看一下,基本都一样:

 

	/**
	 * 配置文件 spring_aop.xml  通过代理
	 * @author 陈丽娜
	 * @version 2015年5月31日上午10:09:10
	 */
	@Test
	public void test(){
		ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop.xml");
		
		Sleepable sleeper =(Sleepable) ct.getBean("linaProxy");
		
		sleeper.sleep();
	}
	
	/**
	 * 配置文件 spring_aop_01.xml   简答的java对象
	 * @author 陈丽娜
	 * @version 2015年5月31日上午10:09:37
	 */
	@Test
	public void test01(){
		ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_01.xml");
		
		Sleepable sleeper = (Sleepable)ct.getBean("lina");
		
		sleeper.sleep();
	}
	
	/**
	 * 配置文件 spring_aop_03.xml  通过aspect注解
	 * @author 陈丽娜
	 * @version 2015年5月31日上午10:09:37
	 */
	@Test
	public void test03(){
		ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_03.xml");
		
        Sleepable sleeper = (Sleepable)ct.getBean("lina");
		
		sleeper.sleep();
	}
	/**
	 * 配置文件 spring_aop_02.xml  通过apsect配置文件
	 * @author 陈丽娜
	 * @version 2015年5月31日上午10:09:37
	 */
	@Test
	public void test02(){
		ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_02.xml");
		
		Sleepable sleeper = (Sleepable)ct.getBean("lina");
		
		sleeper.sleep();
	}
	

 

通过测试类可以看出,不管以什么样的方式来实现aop他们的使用都是没有差别的,这几个测试类的结果都是一样的:


 

作者:jly4758 发表于2015/5/31 17:48:22 原文链接
阅读:0 评论:0 查看评论

相关 [spring aop] 推荐:

Spring AOP详解

- - Java - 编程语言 - ITeye博客
        最近项目中遇到了以下几点需求,仔细思考之后,觉得采用AOP来解决. 一方面是为了以更加灵活的方式来解决问题,另一方面是借此机会深入学习Spring AOP相关的内容. 例如,以下需求不用AOP肯定也能解决,至于是否牵强附会,仁者见仁智者见智. 1.对部分函数的调用进行日志记录,用于观察特定问题在运行过程中的函数调用情况.

Spring AOP监控SQL执行

- - CSDN博客架构设计推荐文章
         对数据库连接池Proxool比较熟悉的读者,都知道Proxool可以记录SQL执行内容和时间等信息日志. 我们可以将该日志记录专门的SQL日志文件,对于查找执行特别耗时的SQL起了不小的作用. 对于一些其他连接池,没有该特性时,本文介绍Spring AOP切面方法来记录SQL日志.

Spring AOP 实现原理与 CGLIB 应用

- - 博客 - 伯乐在线
来源: IBM Developerworks. 简介: AOP(Aspect Orient Programming),也就是面向方面编程,作为面向对象编程的一种补充,专门用于处理系统中分布于各个模块(不同方法)中的交叉关注点的问题,在 Java EE 应用中,常常通过 AOP 来处理一些具有横切性质的系统级服务,如事务管理、安全检查、缓存、对象池管理等.

Spring AOP 代理机制 JDK&CGLIB

- - 开源软件 - ITeye博客
Spring AOP使用JDK动态代理或者CGLIB来为目标对象创建代理. (建议优先使用JDK的动态代理). 如果被代理的目标对象实现了至少一个接口,则会使用JDK动态代理. 所有该目标类型实现的接口都将被代理. 若该目标对象没有实现任何接口,则创建一个CGLIB代理. 如果你希望强制使用CGLIB代理,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法) 那也可以.

使用spring AOP获得session的思路

- - RSS - IT博客云
由于Spring 的AOP面向切面编程,与Servlet容器没有任何关联,所以想要获得Session会话比较麻烦. 当然Struts2同样不依赖Servlet容器,可以在Spring AOP中可以使用 com.opensymphony.xwork2.ActionContext,就可以获得 Session.

Spring AOP + Redis缓存数据库查询

- - 编程语言 - ITeye博客
我们希望能够将数据库查询结果缓存到Redis中,这样在第二次做同样的查询时便可以直接从redis取结果,从而减少数据库读写次数. 必须要做到与业务逻辑代码完全分离. 从缓存中读出的数据必须与数据库中的数据一致. 如何为一个数据库查询结果生成一个唯一的标识. Key),能唯一确定一个查询结果,同一个查询结果,一定能映射到同一个.

Spring AOP动态代理原理与实现方式 (转)

- - 开源软件 - ITeye博客
AOP:面向切面、面向方面、面向接口是一种横切技术. 1.事务管理: (1)数据库事务:(2)编程事务(3)声明事物:Spring AOP-->声明事物   . 3.安全验证: Spring AOP---OOP升级  . 静态代理原理:目标对象:调用业务逻辑    代理对象:日志管理. 表示层调用--->代理对象(日志管理)-->调用目标对象.

基于 Annotation 拦截的 Spring AOP 权限验证方法

- - 企业架构 - ITeye博客
转自: http://www.ibm.com/developerworks/cn/java/j-lo-springaopfilter/index.html. 使用 Annotation 可以非常方便的根据用户的不同角色,分配访问 Java 方法的权限. 在 Java Web 开发中,使用这种方法,可以提高系统的松耦合度,方便维护.

Spring aop 原理及各种应用场景

- - 开源软件 - ITeye博客
AOP是Aspect Oriented Programing的简称,面向切面编程. AOP适合于那些具有横切逻辑的应用:如性能监测,访问控制,事务管理、缓存、对象池管理以及日志记录. AOP将这些分散在各个业务逻辑中的代码通过横向切割的方式抽取到一个独立的模块中. AOP 实现的关键就在于 AOP 框架自动创建的 AOP 代理,AOP 代理则可分为静态代理和动态代理两大类,其中静态代理是指使用 AOP 框架提供的命令进行编译,从而在编译阶段就可生成 AOP 代理类,因此也称为编译时增强;而动态代理则在运行时借助于 JDK 动态代理、CGLIB 等在内存中“临时”生成 AOP 动态代理类,因此也被称为运行时增强.

Spring aop--几种不同的使用方式

- - CSDN博客推荐文章
        最近听老师讲关于AOP关于容器的课,突然觉得之前对aop的理解都不是很到位,先不说理解就是应用都很少,最近也写了几篇关于AOP的博客,都是一些简单的demo,今天在这里再说一下关于Spring AOP的几种实现方式.         我们经常会用到的有如下几种.         1、基于代理的AOP.