Spring 通过注解方式实现AOP切面

4 篇文章 0 订阅

        Spring 切面编程的目的是实现代码的业务逻辑的解耦。切面编程用于诸如日志记录,事务处理,等非业务性的逻辑操作。目前Spring的Aop只能应用于方法层级上,无法在类、成员字段等层级上操作。以下是Srping的Aop编程分为注解方式和xml配置方式。以下过程详细说明了通过注解方式实现AOP编程的过程。

第一步:自定义注解


 
 
  1. /**
  2. * 定义自定义管理员注解
  3. *
  4. */
  5. public @interface RoleAdmin {
  6. String values() default "";
  7. String descript() defalut "自定义管理员注解";
  8. }

  • 第二步:编写AOP切面类


 
 
  1. /*声明为组件,让spring 自动管理*/
  2. @Component
  3. /*声明该类为切面bean*/
  4. @Aspect
  5. public class RequestInterceptor {
  6. private final Logger logger = LoggerFactory.getLogger(RequestInterceptor.class);
  7. /* 拦截管理员访问的请求,(定义切点,该方法无方法体,主要为方便同类中其他方法使用此处配置的切入点)*/  
  8. @Pointcut( "@annotation(com.frame.annotation.RoleAdmin)")
  9. public void adminRequired() {}
  10. /*定义前置advice,同时接受JoinPoint切入点对象,可以没有该参数
  11. 在环绕通知里面proceedingJoinPoint参数是必须的,其他情况JoinPoint 并不是必须的
  12. */
  13. @Before( "adminRequired()")
  14. public void adminCommon(JoinPoint point) {       
  15. HttpServletRequest request = getRequest();
  16.         LoginUser user = (LoginUser) request.getSession().getAttribute(Constants.SESSION_USER_KEY);
  17.          if ( user == null ) {
  18.             setForward(request);
  19.              throw new NoAdminException();
  20.              //throw new NotLoginException();
  21.         }
  22.          if  ( user.getMemberType() != Constants.MEMBER_TYPE_ADMIN ) {
  23.              throw new NoAdminException();
  24.              //throw new NoAuthException();
  25.         }
  26.         String mac = (String)request.getSession().getAttribute(AdminUtil.ADMIN_MAC_ATTR_NAME);
  27.          if(UtilString.isEmpty(mac)){
  28.              throw new NoAdminException();
  29.         }
  30. }
  31. private void setForward(HttpServletRequest request) {
  32. if ( "get".equalsIgnoreCase(request.getMethod()) ) {
  33. String fw = request.getRequestURI().substring(request.getContextPath().length()+ 1);
  34. String qs = request.getQueryString();
  35. if ( ! UtilString.isEmpty(qs) ) {
  36. fw += "?" + qs;
  37. }
  38. logger.debug( "After login will forward to : {}", fw);
  39. request.getSession().setAttribute(Constants.URL_FORWARD_KEY, fw);
  40. }
  41. }
  42. /** * 在切面中获取http请求 * @return */
  43. private HttpServletRequest getRequest() {
  44. return ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
  45. }
  46. }
 

  • 第三步:在application.xml中配置aop代理

               需要在application.xml中添加AOP代理。AOP代理分为JDK代理和cglib代理

               如果配置proxy-target-class="true",则表示是cglib代理。如下:


 
 
  1. <!-- 激活自动代理功能 -->
  2. <aop:aspectj-autoproxy proxy-target-class="true"/>

               如果采用JDK代理,则配置如下:

       <aop:aspectj-autoproxy/>
 
 

  •  第四步: 定义切面的引入点


 
 
  1. /*在需要切面的方法上定义JoinCut点*/
  2. @RoleAdmin(descript="在这里定义")
  3. @ResponseBody
  4. @RequestMapping(value="admin/activity/updateComment")
  5. public boolean updateCommentIsDelete(HttpServletRequest request,CommentPlan commentPlan){
  6. return false;
  7. }

        切入点匹配多个连接点的两种方式

  • 通过@annotation精确切入到某个自定义标签中

       @Pointcut("@annotation(com.frame.annotation.RoleAdmin)")
 
 

  • 通过exrcution模糊匹配

       @Pointcut("execution(* cn.ysh.studio.spring.aop.service..*(..))")  
 
 

              execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)

  • modifiers-pattern:方法的操作权限
  • ret-type-pattern:返回值
  • declaring-type-pattern:方法所在的包
  • name-pattern:方法名
  • parm-pattern:参数名
  • throws-pattern:异常

         其中,除ret-type-pattern和name-pattern之外,其他都是可选的。上例中,execution(* com.spring.service.*.*(..))表示com.spring.service包下,返回值为任意类型;方法名任意;参数不作限制的所有方法。

         AOP获取注解和Target对象


 
 
  1. /*定义前置advice*/
  2. @Before( "execution(* cn.ysh.studio.spring.aop.service..*(..))&&@annotation(roleAdmin)")
  3. public void adminCommon(JoinPoint point, RoleAdmin roleAdmin) {
  4. /*输出注解描述内容*/
  5. System.out.println(roleAdmin.descript());
  6. /*利用反射显示目标对象的相关信息*/
  7. String targetName = joinPoint.getTarget().getClass().getName();   
  8.       String methodName = joinPoint.getSignature().getName();    
  9.       Object[] arguments = joinPoint.getArgs();   
  10.       Class targetClass = Class.forName(targetName);    
  11.       Method[] methods = targetClass.getMethods();   
  12.       String description = "";
  13.       for(Method method : methods) {   
  14.       if(method.getName().equals(methodName)) {    
  15.       Class[] clazzs = method.getParameterTypes();    
  16.         if (clazzs.length == arguments.length) {   
  17.         description = method.getAnnotation(SystemControllerLog.class).description();    
  18.           break;    
  19.         }   
  20.       } 
  21. }
  22.  

 

         在Spring中,任何通知(Advice)方法都可以将第一个参数定义为 org.aspectj.lang.JoinPoint类型用以接受当前连接点对象。JoinPoint接口提供了一系列有用的方法, 比如 getArgs() (返回方法参数)、getThis() (返回代理对象)、getTarget() (返回目标)、getSignature() (返回正在被通知的方法相关信息)和 toString() (打印出正在被通知的方法的有用信息)。

        1、如果要设置多个切点,则使用||进行拼接

    @Pointcut("execution(* aop.annotation.*.*(..))|| execution(*com.action.admin.*.*update*(..))")
 
 

        2、环绕通知的方法中一定要有ProceedingJoinPoint 参数,与Filter中的doFilter方法类似)

 
 
  1. @Around( "execution(* aop.annotation.*.*(..))")
  2. public Object doAround(ProceedingJoinPoint pjp) throws{
  3. }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值