`
saviorzuo
  • 浏览: 19774 次
  • 性别: Icon_minigender_1
  • 来自: 上海
最近访客 更多访客>>
社区版块
存档分类
最新评论

spring AOP(2)

阅读更多
前面写过一篇关于Spring AOP方面的文章,探讨了Spring AOP底层实现的一些细节知识,这里面涉及到了JAVA反射机制,代理模式以及CGLIB库的使用。也就是说,Spring AOP底层实现就是靠动态代理(针对有接口的类)和CGLIB(针对没有实现接口的一般类),那么,有了这些知识,再辅佐对核心配置XML文件解析的能力,其实就可以实现一个简易的基于IOC和AOP的小框架,大家可以自己尝试着写一下。下面呢我们就由浅入深地来看看在Spring中AOP是怎么实现的。

       最简单的AOP实现只需要涉及3个概念:目标(Target),通知(Advice)和代理(Proxy)。目标呢,当然就是真实的需要被代理的对象,一般它会实现至少一个接口。通知呢,就是当目标的方法调用时需要调用的代码,也叫拦截器。而代理,毫无疑问就是加入了通知的目标了,它可以作为目标的替身出现。为了说明这三者的关系,我们来看一个网上有趣的小例子:一间书店开始打折促销,规则是每一名顾客只能买一本书,并且当顾客来到书店时,要说喜欢您来。顾客走的时候,还要说喜欢您再来!(麦当劳啊^_^) 顾客如果买到<hibernate in action>这本书,要抛出异常,告知他没有存货!呵呵,好啦,洪哥,我们动手吧!
package com.wepull.spring.book;
 
public class NoThisBookException extends Exception {
 
       public NoThisBookException(String msg) {
              super(msg);
       }
 
}
 
package com.wepull.spring.book;
 
public interface BuyBook {
       public void buyBook(String customer,String book) throws NoThisBookException;
}
 
package com.wepull.spring.book;
 
public class MyBuyBook implements BuyBook{
 
       public void buyBook(String customer, String book)
                     throws NoThisBookException {
              if(book.equals("<hibernate in action>"))
              throw new NoThisBookException("对不起,没有"+book+"的存货了!");
       System.out.println(customer+",你好,你已经购买了一本"+book+"!");
             
       }
      
}


看上面,我们做了一个异常类和一个简单买书的接口,并且目标MyBuyBook也已经出现。OK,再来看看通知吧!

Spring中主要有以下四种通知类型:
(1)Around通知
(2)Before通知
(3)Throws通知
(4)After Returning通知

例子分别如下:
package com.wepull.spring.book;
 
import java.util.HashSet;
import java.util.Set;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
 
public class MyAroundAdvice implements MethodInterceptor {
    private Set customers=new HashSet(); //保存购过书的顾客信息
 
    public Object invoke(MethodInvocation invocation) throws Throwable {
        Object[] args= invocation.getArguments();
        if(customers.contains(args[0])){
            System.out.println("对不起,一名顾客只能买一本打折书!");
            return null;
        }
        customers.add(args[0]);
        return invocation.proceed();
    }
 
}
package com.wepull.spring.book;
 
import java.lang.reflect.Method;
import org.springframework.aop.MethodBeforeAdvice;
 
public class MyBeforeAdvice implements MethodBeforeAdvice {
 
    public void before(Method arg0, Object[] arg1, Object arg2)
            throws Throwable {
        String customer=(String)arg1[0]; //第2个参数组就是被通知的方法传入的参数,本例中即customer,book
        System.out.println("喜欢您来!"+customer+"!"); //显示欢迎信息!,在buyBook方法前调用
 
 
    }
 
}
package com.wepull.spring.book;
 
import java.lang.reflect.Method;
import org.springframework.aop.ThrowsAdvice;
 
public class MyThrowsAdvice implements ThrowsAdvice {
 
    public void afterThrowing(Method method, Object[] args, Object target, NoThisBookException e){
        //可以定义多个方法,只要传入的参数是不同异常
        System.out.println("对不起"+args[0]+",没货了。通知仓库,赶紧加书!");
 
    }
}
package com.wepull.spring.book;
 
import java.lang.reflect.Method;
import org.springframework.aop.AfterReturningAdvice;
 
public class MyAfterAdvice implements AfterReturningAdvice {
 
    public void afterReturning(Object arg0, Method arg1, Object[] arg2,
            Object arg3) throws Throwable {
        String customer=(String)arg2[0]; //同前置通知一样,参数组3为传入参数,具体见spring doc
        System.out.println("喜欢您再来!"+customer+"!"); //显示欢送信息!
 
 
    }
 
}


现在目标类有了,通知类也写好了,写了就要配,这是框架的基本使用原则。呵呵,下面就来看看核心配置文件src/applicationContext.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"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
    <!-- 各种通知 -->
    <!-- 前置通知 -->
    <bean id="myBeforeAdvice"
        class="com.wepull.spring.book.MyBeforeAdvice">
    </bean>
    <!-- 后置通知 -->
    <bean id="myAfterAdvice"
        class="com.wepull.spring.book.MyAfterAdvice">
    </bean>
    <!-- 异常通知 -->
    <bean id="myThrowsAdvice"
        class="com.wepull.spring.book.MyThrowsAdvice">
    </bean>
    <!-- 环绕通知 -->
    <bean id="myAroundAdvice"
        class="com.wepull.spring.book.MyAroundAdvice">
    </bean>
    <!-- 目标对象 -->
    <bean id="buyBookTarget"
        class="com.wepull.spring.book.MyBuyBook">
    </bean>
    <!-- 代理对象 -->
<!-- ProxyFactoryBean被用于创建代理 -->
    <bean id="buyBook"
        class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 实现的接口 -->
        <property name="proxyInterfaces">
            <value>com.wepull.spring.book.BuyBook</value>
        </property>
        <!-- 应用所有 (通知)拦截器对象到所有调用 -->
        <property name="interceptorNames">
            <list>
                <value>myBeforeAdvice</value>
                <value>myAfterAdvice</value>
                <value>myThrowsAdvice</value>
                <value>myAroundAdvice</value>
            </list>
        </property>
        <!-- 代理的目标对象 -->
        <property name="target">
            <ref bean="buyBookTarget" />
        </property>
    </bean>
</beans>


注意上面的ProxyFactoryBean类是在BeanFactory中显式地创建代理对象的中心类。你给它一个要实现的接口,一个要代理的目标对象,若干个要织入的通知,它就会创建一个崭新的代理对象。所以通常配置ProxyFactoryBean,让它实现和目标对象一样的接口。最后当然是测试代码了。
package com.wepull.spring.book;
 
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
 
public class Test {
 
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        BuyBook buyBook = (BuyBook) context.getBean("buyBook");
        try {
            buyBook.buyBook("leno", "<struts in action>");
            buyBook.buyBook("leno", "<spring in action>");
            buyBook.buyBook("lenotang", "<hibernate in action>");
        } catch (NoThisBookException e) {
            e.printStackTrace();
        }
    }
}


这就是Spring中最简单的AOP应用了。怎么样,并不是很难吧。大家已经看到,如果我们的接口里面定义了很多方法,那么所有的方法都会置入这些通知,但有时候我们是希望有所选择的,怎么改进呢?还有,如果我们还有很多个这样需要置入这些通知的对象存在,那针对每一个目标都要做一个几乎一样的代理的配置,似乎也欠妥。那么,如何优化呢。这就是我们下一篇文章的主题了。呵呵,继续前进吧!

[转]http://blog.csdn.net/lenotang/archive/2008/10/07/3030753.aspx

分享到:
评论

相关推荐

    spring aop jar 包

    Spring AOP(Aspect Oriented Programming,面向切面编程)是Spring框架的重要组成部分,它提供了一种在不修改源代码的情况下,对程序进行功能增强的技术。这个"spring aop jar 包"包含了实现这一功能所需的类和接口,...

    Spring AOP 16道面试题及答案.docx

    Spring AOP,全称为Aspect Oriented Programming,是面向切面编程的一种编程范式,它是对传统的面向对象编程(OOP)的一种补充。在OOP中,核心是对象,而在AOP中,核心则是切面。切面是关注点的模块化,即程序中的...

    简单spring aop 例子

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种模块化和声明式的方式来处理系统中的交叉关注点问题,如日志、事务管理、安全性等。本示例将简要介绍如何在Spring应用中实现AOP,通过实际的...

    Spring AOP完整例子

    Spring AOP(面向切面编程)是Spring框架的核心特性之一,它允许开发者在不修改源代码的情况下,通过插入切面来增强或改变程序的行为。在本教程中,我们将深入探讨Spring AOP的不同使用方法,包括定义切点、通知类型...

    spring aop依赖jar包

    现在,我们回到主题——"springaop依赖的jar包"。在Spring 2.5.6版本中,使用Spring AOP通常需要以下核心jar包: - `spring-aop.jar`:这是Spring AOP的核心库,包含了AOP相关的类和接口。 - `spring-beans.jar`:...

    spring-aop.jar各个版本

    spring-aop-1.1.1.jar spring-aop-1.2.6.jar spring-aop-1.2.9.jar spring-aop-2.0.2.jar spring-aop-2.0.6.jar spring-aop-2.0.7.jar spring-aop-2.0.8.jar spring-aop-2.0.jar spring-aop-2.5.1.jar spring-aop-...

    Spring Aop四个依赖的Jar包

    Spring AOP,全称Aspect-Oriented Programming(面向切面编程),是Spring框架的一个重要模块,它通过提供声明式的方式来实现面向切面编程,从而简化了应用程序的开发和维护。在Spring AOP中,我们无需深入到每个...

    死磕Spring之AOP篇 - Spring AOP两种代理对象的拦截处理(csdn)————程序.pdf

    Spring AOP 是一种面向切面编程的技术,它允许我们在不修改源代码的情况下,对应用程序的特定部分(如方法调用)进行增强。在 Spring 中,AOP 的实现主要依赖于代理模式,有两种代理方式:JDK 动态代理和 CGLIB 动态...

    Spring AOP面向方面编程原理:AOP概念

    2. **轻量级**:与一些需要预编译器的AOP框架不同,Spring AOP无需特殊的工具或编译步骤即可使用。这种轻量级的特性使得Spring AOP更易于学习和集成。 3. **灵活的通知模型**:Spring AOP提供了多种类型的通知,...

    spring aop 自定义注解保存操作日志到mysql数据库 源码

    2、能够清楚的知道如何用spring aop实现自定义注解以及注解的逻辑实现 (需要知道原理的请看spring aop源码,此处不做赘述) 3、可在现有源码上快速进行功能扩展 4、spring boot,mybatis,druid,spring aop的使用

    Spring AOP实现机制

    **Spring AOP 实现机制详解** Spring AOP(面向切面编程)是Spring框架的核心特性之一,它允许程序员在不修改源代码的情况下,通过“切面”来插入额外的业务逻辑,如日志、事务管理等。AOP的引入极大地提高了代码的...

    基于注解实现SpringAop

    基于注解实现SpringAop基于注解实现SpringAop基于注解实现SpringAop

    spring aop的demo

    在`springAop1`这个压缩包中,可能包含了一个简单的应用示例,展示了如何定义一个切面类,以及如何在该类中定义通知方法。例如,我们可能会看到一个名为`LoggingAspect`的类,其中包含了`@Before`注解的方法,用于在...

    spring aop 五个依赖jar

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种模块化和声明式的方式来处理系统中的交叉关注点,如日志、事务管理等。在Java应用中,AOP通过代理模式实现了切面编程,使得我们可以将业务逻辑...

    spring AOP依赖三个jar包

    Spring AOP,即Spring的面向切面编程模块,是Spring框架的重要组成部分,它允许开发者在不修改源代码的情况下,对程序进行横切关注点的处理,如日志、事务管理等。实现这一功能,主要依赖于三个核心的jar包:aop...

    spring AOP 引入jar包,spring IOC 引入Jar包

    Spring AOP 和 Spring IOC 是 Spring 框架的两个核心组件,它们对于任何基于 Java 的企业级应用开发都至关重要。Spring AOP(面向切面编程)允许开发者在不修改源代码的情况下,通过“切面”来插入新的行为或增强已...

    @AspectJ配置Spring AOP,demo

    `springAOP2`可能是一个包含具体示例代码的目录。`基于@AspectJ配置Spring AOP之一 - 飞扬部落编程仓库-专注编程,网站,专业技术.htm`和其关联的`_files`目录可能包含了一个详细的教程或演示如何配置和运行@AspectJ的...

    反射实现 AOP 动态代理模式(Spring AOP 的实现原理)

    面向切面编程(AOP)是一种编程范式,旨在将横切关注点(如日志、安全等)与业务逻辑分离,从而提高模块化。...利用Java反射机制和Spring AOP框架,开发者可以方便地实现AOP,从而提升代码的模块化和可维护性。

    spring-aop实例

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种强大的方式来实现横切关注点,如日志、事务管理、安全性等,从而解耦应用程序的核心业务逻辑。在Spring AOP中,关注点被模块化为独立的“切面”...

    spring aop 学习笔记

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种模块化和抽象化的方法来处理系统中的交叉关注点,如日志、事务管理、安全性等。本学习笔记将深入探讨Spring AOP的核心概念、工作原理以及实际...

Global site tag (gtag.js) - Google Analytics