文档章节

spring 拦截器(转)

~少司命~
 ~少司命~
发布于 2012/12/14 10:32
字数 1306
阅读 451
收藏 0

阿里云携手百名商业领袖、技术大咖,带您一探行进中的数字新基建!>>>


 

接触spring有一段时间了,不过都是看的多,写的少,工作忙,公司也不准备用。自己写过一些小东西用到,也只用到了BeanFactory组装对 象,JdbcTemplate代替jdbc,事务管理。。。东抓一把,西抓一把,没形成系统。最近也在看spring自带的reference,一时手 痒,写了个AOP的创建advice的例子,比之单纯地使用TransationProxyFactoryBean对AOP的理解又深入了点,打算看看它的源代码,基于CGLIB的实现对类的代理不了解,倒是好奇它如何实现对接口的代理——也就是利用J2SE的动态代理技术。

例子如下:讲述一间书店开始打折促销,规则是每一名顾客只能买一本书,而且<spring in action>已经脱销了。。。。。你可以去掉TestAdvice里的注释看看各种运行结果,具体就不解释咯,在代码注释里。

首先,你必须对增强(advice)有所了解,增强就是在特定连接点执行的动作。advice contains the logic of your aspect。增强,分为4类:

  1. 前增强(before) 在连接点之前调用
  2. 后增强(after) 在连接点执行之后调用、
  3. 环绕增强(around) 完整控制整个方法流程,必须调用MethodInvocation的proceed促使真实操作发生
  4. 异常增强 针对某个异常抛出时调用

书店,一个buyBook业务:

package com.denny_blue.springdemo.aop;

public interface BuyBook {
      public void buyBook(String customer,String book)throws NoThisBookException;
}

实现此接口的一个业务对象,如果顾客要买<spring in action>就抛出NoThisBookException异常。

package com.denny_blue.springdemo.aop;

public class MyBuyBook implements BuyBook {
 public void buyBook(String customer,String book)throws NoThisBookException{
  if(book.equals("<spring in action>"))
   throw new NoThisBookException("对不起,没有"+book+"存货了!");
  System.out.println(customer+",你好,你已经购买了一本"+book+"!");
 }
}

自定义两个异常类,其中NoThisBookException被抛出时将触发MyThrowsAdvice调用:

//NoThisBookException.java

package com.denny_blue.springdemo.aop;


public class NoThisBookException extends RuntimeException {
 public NoThisBookException(String msg){
  super(msg);
 }

}

//BuyBookException

package com.denny_blue.springdemo.aop;

public class BuyBookException extends RuntimeException {
 public BuyBookException(String msg){
  super(msg);
  System.out.println(msg);
 }

}

OK,接下来就是各类增强了,前增强,在业务方法buyBook之前调用,显示欢迎信息:

package com.denny_blue.springdemo.aop;

import java.lang.reflect.Method;

import org.springframework.aop.MethodBeforeAdvice;

public class MyBeforeAdvice implements MethodBeforeAdvice {


 public void before(Method arg0, Object[] arg1, Object target)
   throws Throwable {
  String customer=(String)arg1[0];     //第2个参数组就是被增强的方法传入的参数,本例中即customer,book
  System.out.println("欢迎光临!"+customer+"!");  //显示欢迎信息!,在buyBook方法前调用

 }

}

 

然后是后增强,当顾客已经买了书之后,显示欢送信息:

package com.denny_blue.springdemo.aop;

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+"!");  //显示欢送信息!
 }

}

OK,有了上面两个advice我们就能提供给顾客很好的服务态度了,等等?我们还有规则没实现,不是说一名顾客只能买一本书吗?OK,我们用环绕增强来解决,在环绕增强中保存一个HashSet,判断顾客是否来过,来过就抛出一个异常,没来过再放入此Set中:

package com.denny_blue.springdemo.aop;

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 {
       String customer=(String)invocation.getArguments()[0];
       if(customers.contains(customer)){
        throw new BuyBookException("对不起,一名顾客只能买一本打折书!");
       }
       Object result=invocation.proceed();  //调用MyBuyBook中的buyBook方法,即真实操作
       customers.add(customer);
       return result;
 }

}

好了,最后一个,异常增强,当顾客要买<spring in action>时,我们的书店没书了,请仓库部门赶快订货!!

package com.denny_blue.springdemo.aop;

import org.springframework.aop.ThrowsAdvice;

public class MyThrowsAdvice implements ThrowsAdvice {
   public void afterThrowing(NoThisBookException e){   //可以定义多个方法,只要传入的参数是不同异常
     System.out.print("通知仓库,赶紧加订书!");
   }

}

好了没?还没,我们需要一个XML文件来组装这些对象,来代理业务接口,完整的beans.xml如下,各元素的含义请自己查看spring reference

?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">

<beans>
    <bean id="myBuyBookTarget" class="com.denny_blue.springdemo.aop.MyBuyBook"/>
    <bean id="myBeforeAdvice" class="com.denny_blue.springdemo.aop.MyBeforeAdvice"/>
    <bean id="myAroundAdvice" class="com.denny_blue.springdemo.aop.MyAroundAdvice"/>
    <bean id="myAfterAdvice" class="com.denny_blue.springdemo.aop.MyAfterAdvice"/>
    <bean id="myThrowsAdvice" class="com.denny_blue.springdemo.aop.MyThrowsAdvice"/>
    <bean id="buyBook" class="org.springframework.aop.framework.ProxyFactoryBean">
       <property name="proxyInterfaces">
           <value>com.denny_blue.springdemo.aop.BuyBook</value>
       </property>
       <property name="interceptorNames">
           <list>
                 <value>myBeforeAdvice</value>
                 <value>myAfterAdvice</value>
                 <value>myAroundAdvice</value>
                 <value>myThrowsAdvice</value>
           </list>
       </property>
       <property name="target">
           <ref bean="myBuyBookTarget"/>
       </property>
    </bean>
</beans>

我们先声明所有的bean,通过

 <bean id="myBuyBookTarget" class="com.denny_blue.springdemo.aop.MyBuyBook"/>
确定将要被增强的对象——目标对象(target),我们可以很容易地替换这个目标对象,只要它实现业务接口。代理的接口通过:

<property name="proxyInterfaces">
           <value>com.denny_blue.springdemo.aop.BuyBook</value>
</property>
设定,然后是要用到一系列增强,注意,顺序是很有影响的!你可以尝试着改变顺序看看结果:)

<property name="interceptorNames">
           <list>
                 <value>myBeforeAdvice</value>
                 <value>myAfterAdvice</value>
                 <value>myAroundAdvice</value>
                 <value>myThrowsAdvice</value>
           </list>
</property>

一切准备好了,我们来测试吧,GO GO GO

package test;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.FileSystemXmlApplicationContext;

import com.denny_blue.springdemo.aop.BuyBook;

public class TestAdvice {
 public static void main(String args[]){
  ApplicationContext ctx=new FileSystemXmlApplicationContext("/src/beans.xml"); 

 //我的beans.xml放在项目下的src目录,eclipse环境下,请自己调整
  BuyBook buybook=(BuyBook)ctx.getBean("buyBook");
  buybook.buyBook("jordan","<深入浅出hibernate>");
 // buybook.buyBook("dennis","<spring in action>");   //去掉注释即可观察异常增强
 // buybook.buyBook("jordan","<深入浅出hibernate>");  //去掉注释即可观察环绕增强

 }
}

本文转载自网络

~少司命~
粉丝 3
博文 76
码字总数 8735
作品 0
昌平
高级程序员
私信 提问
加载中

评论(0)

SpringBoot中的拦截机制

SpringBoot的拦截机制实现不是其特有的,它和Spring中一致,都可以使用下面三种技术来实现拦截机制 过滤器(Filter) 拦截器(Interceptor) 切片(Aspect) 过滤器拦截 过滤器是servlet中的...

Coding小聪
2018/04/06
0
0
Spring mvc 拦截器拦截后跳转方法疑惑

Spring mvc 拦截器拦截后方法疑惑: 网上Spring mvc 拦截器拦截后跳转有两种方法: (1)response重定向(传信息很不方便,get方式将信息写进去) response.sendRedirect("/Go/login/login.j...

李东亮
2014/07/30
5.3K
0
spring配置的拦截器不执行

一直在工作中使用的springmvc,最近跟遇上一个奇怪的问题,在项目中通过mvc:interceptor的方式注入的拦截器不被执行。下面是配置文件: web.xml spring-servlet.xml applicationContext.xml s...

回忆若风
2016/07/19
3.3K
4
HandlerMethodArgumentResolver用于统一获取当前登录用户 — 转

出处: HandlerMethodArgumentResolver用于统一获取当前登录用户 目录   一、最原始直接   二、AOP   三、拦截器+方法参数解析器     3.1 自定义权限拦截器      3.2 自定义参...

myseries
05/02
0
0
Spring MVC中的拦截器/过滤器HandlerInterceptorAdapter的使用

一般情况下,对来自浏览器的请求的拦截,是利用Filter实现的 而在Spring中,基于Filter这种方式可以实现Bean预处理、后处理。 比如注入FilterRegistrationBean,然后在这个Bean上传递自己继承...

xiaoshen666
2019/08/11
0
0

没有更多内容

加载失败,请刷新页面

加载更多

SpringBoot 整合 Redis 缓存

1.首先导入使用Maven导入jar包 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId></dependency><......

FH-Admin
28分钟前
12
0
如何安装WordPress插件 - 初学者的分步指南 - WP站长

<!-- wp:paragraph -->安装WordPress后,每一个初学者需要学习的第一件事就是如何安装WordPress插件。插件允许您向WordPress添加新功能,例如添加图库、幻灯片等。有数千个可用于WordPress的...

wpzhanzhang
43分钟前
8
0
【Flutter组件终结篇】332个组件 658页PDF

老孟导读:历时1年的时间,整理完成了330+组件的详细用法,不仅包含UI组件,还包含了功能性的组件。 虽然整理了 330+的组件基本用法,但并不是让你每一个都学习一遍,任何技术基本都是掌握 ...

老孟Flutter
今天
17
0
三星手机又中招:一张壁纸可引发系统崩溃 附临时解决方法

  前几天国内有大量用户发现三星手机崩溃、黑屏或者无限重启, 这可能是三星手机的日历 APP 的 bug。这件事还没完,三星手机今天又发现了新的问题,换上一张特别的壁纸就会导致系统崩溃,不...

alkcendkljk
今天
13
0
查找当前目录和文件目录[重复] - Find current directory and file's directory [duplicate]

问题: This question already has answers here : 这个问题已经在这里有了答案 : How to properly determine current script directory? 如何正确确定当前脚本目录? (11 answers) (11个答...

技术盛宴
今天
27
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部