在《初识Spring》中我们了解到Spring支持AOP且可配置方法的前置曾强和后置曾强,但其实Spring支持多种曾强类型。下面同过一些例子来介绍Spring的几种常用的曾强(前置增强和后置曾强不再进行介绍,详情可参阅《初识Spring》这篇博客)。

异常抛出曾强

异常抛出曾强的特点是在目标方法抛出异常时织入曾强处理。首先我们要编写一个实现异常曾强代码的类,给类实现ThrowsAdvice接口。如下所示:

package cn.wz.aop;

import java.lang.reflect.Method;

import org.apache.log4j.Logger;
import org.springframework.aop.ThrowsAdvice;
public class ErrorLogger implements ThrowsAdvice {
Logger log=Logger.getLogger(ErrorLogger.class);
/**
* 实现异常曾强代码的方法
* @param method 目标方法名
* @param args 向目标方法传入的参数
* @param target 目标方法所在的类的实例
* @param e 目标方法内所抛出的异常对象
*/
public void afterThrowing(Method method,Object[] args,Object target,Exception e) {
log.error(method.getName()+"发生异常:"+e);
}
}

 

上述代码通过实现ThrowsAdvice接口实现异常抛出曾强,其中ThrowsAdvice接口中没有定义任何方法,但我们在定义异常抛出的曾强方法时必须遵守以下方法签名:

void afterThrowing([Method method,Object [] arguments,Object target,]Throwable ex)

这里Spring规定了方法名必须是“afterThrowing”。方法参数只有最后一个是必须的,前面三个参数是可选的,但是前面三个参数只能是要么都提供,要么都不提供!否则则无法实现曾强。下面编写测试类进行测试:

package cn.wz.test;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class ExceptionTest {
public static void main(String[] args) throws Exception {
ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");
ExceptionTest bean = context.getBean("test",ExceptionTest.class);
bean.add();
} public void add() throws Exception {
throw new Exception("错误"); }
}

最终运行结果:

对于Spring配置文件的配置方法与前置增强和后置曾强的配置方法相同这里不再进行演示,

环绕曾强

环绕曾强在目标方法的前后都可以织入曾强处理。环绕增强是功能最强大的增强处理,Spring把目标方法的控制权全部交给了它。在环绕曾强处理中,可以获取或修改目标方法的参数,返回值,可以对它进行异常处理,甚至可以决定目标方法是否执行!

好了下面来看看如何实现环绕曾强吧。首先还是要定义实现环绕曾强代码的类,该类实现了MethodInterceptor接口的invoke()方法,在invoke()方法中编写曾强代码。

package cn.wz.aop;

import java.lang.reflect.Method;
import java.util.Arrays; import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
import org.apache.log4j.Logger;
public class ErrorLogger implements MethodInterceptor {
Logger log=Logger.getLogger(ErrorLogger.class); public Object invoke(MethodInvocation arg0) throws Throwable {
Method method = arg0.getMethod();
Object[] arguments = arg0.getArguments();
Object target = arg0.getThis();
log.info("调用"+target+"的"+method.getName()+"方法。方法入参:"+Arrays.toString(arguments)); try {
Object result = arg0.proceed();
log.info("调用"+target+"的"+method.getName()+"方法。方法返回值:"+result);
return result;
} catch (Exception e) {
log.error(method.getName()+"方法异常:"+e);
}
return null;
}
}

上述代码通过MethodInterceptor接口实现了环绕增强。该接口要求实现invoke方法,其参数MethodInvocation不但疯转了目标方法及其入参数组,还封装了被代理的目标对象。通过proceed方法可以调用目标对象响应的方法,从而实现对目标方法的完全控制。借助异常抛出曾强的测试代码进行测试其运行结果如下:

使用注解实现曾强

除了实现Spring提供的特定接口外,Spring还可以通过集成AspectJ实现了以注解的方式定义增强类。大大减少了配置文件中的工作量。

注意:使用Aspectj是要确保JDK的版本是5.0或以上的版本,否则将无法使用注解技术,其次还要引入asm模块的jar包到项目中

前置曾强和后置曾强类:

package cn.wz.aop;

import org.apache.log4j.Logger;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before; @Aspect
public class UserLogger {
private static final Logger log=Logger.getLogger(UserLogger.class);
@Before("execution(public void add())")
public void before(){
log.info("前置增强");
}
@AfterReturning("execution(public void add())")
public void afterReturning(){
log.info("后置增强");
}
}

在Spring配置文件中的配置如下:

异常抛出增强类:

package cn.wz.aop;

import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect; @Aspect
public class Error_Logger {
private static final Logger log=Logger.getLogger(UserLogger.class);
@AfterThrowing(pointcut="execution(public void add())" ,throwing="e")
public void agterThrowing(JoinPoint jp,Exception e){
log.error(jp.getSignature().getName()+"发生异常:"+e);
}
}

环绕增强类:

package cn.wz.aop;

import org.apache.log4j.Logger;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
@Aspect
public class ArounsLogger {
private static final Logger log=Logger.getLogger(UserLogger.class);
@Around("execution(public void add())")
public Object aroundLogger(ProceedingJoinPoint jp) throws Throwable{
System.out.println("前置增强");
Object obj = jp.proceed();//调用原始方法
System.out.println("后置增强");
return obj; }
}

最新文章

  1. Python全栈开发day7
  2. oracle 数据库导出数据
  3. 说一说vector<bool>
  4. HDU 3255 扫描线(立方体体积并变形)
  5. 纯CSS实现的右侧底部简洁悬浮效果
  6. homework-09
  7. EasyUI combotree值的设置 setValue
  8. A Tour of Go Struct Fields
  9. easyui的样式easyui-textbox的一个bug
  10. 15-UIKit(view布局、Autoresizing)
  11. IOS开发-UI学习-UISlider(滑动条)的使用
  12. cookie原理
  13. Oracle学习笔记_09_字符串相关函数
  14. linux Centos7下安装python3及pip3
  15. L2-010 排座位 (25 分) (最短路)
  16. centos 7 开机yum
  17. 在JAVA中对于类,对象,继承,多态的看法
  18. MySQL 之 数据操作
  19. Kafka 0.11版本新功能介绍 —— 空消费组延时rebalance
  20. python学习笔记——多进程一 基础概念

热门文章

  1. JQuery图片轮播滚动效果(网页效果--每日一更)
  2. 浅谈Excel开发:六 Excel 异步自定义函数
  3. 使用protractor操作页面元素
  4. Weibo用户地图
  5. git 仓库、分支的区别
  6. 说说设计模式~门面模式(Facade)
  7. 大叔也说Xamarin~Android篇~原生登陆与WebView的网站如何共享Session
  8. 02- Shell脚本学习--运算符
  9. JS 操作 DOM
  10. spring之BeanFactory