Spring Aop通知(基于@AspectJ注解实现)

    实现特定功能的方面代码在AOP概念中称为“通知(Advice)”。通知分为前置通知、后置通知、环绕通知和异常通知。

    这个分类是依据通知织入到业务代码时执行的时间划分的:

    前置通知:在方法执行之前自动执行的通知。

    后置通知:在方法执行之后自动执行的通知。

    环绕通知:可以在方法调用前执行通知代码,可以决定是否调用目标方法。

    异常通知:在方法抛出异常时自动执行方面代码。

项目结构


log4j.properties

log4j.rootLogger=info,stdout,info,debug,error
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=[%-5p] [%d{HH:mm:ss}] %c - %m%n
 
log4j.logger.info=info
log4j.appender.info=org.apache.log4j.DailyRollingFileAppender
log4j.appender.info.layout=org.apache.log4j.PatternLayout
log4j.appender.info.layout.ConversionPattern=[%-5p] [%d{HH:mm:ss}] %c - %m%n
log4j.appender.info.datePattern='.'yyyy-MM-dd
log4j.appender.info.Threshold = INFO 
log4j.appender.info.append=true
log4j.appender.info.File=${webApp.root}/WEB-INF/logs/info.log
 
log4j.logger.debug=debug
log4j.appender.debug=org.apache.log4j.DailyRollingFileAppender
log4j.appender.debug.layout=org.apache.log4j.PatternLayout
log4j.appender.debug.layout.ConversionPattern=[%-5p] [%d{HH:mm:ss}] %c - %m%n
log4j.appender.debug.datePattern='.'yyyy-MM-dd
log4j.appender.debug.Threshold = DEBUG 
log4j.appender.debug.append=true
log4j.appender.debug.File=${webApp.root}/WEB-INF/logs/debug.log

log4j.logger.error=error
log4j.appender.error=org.apache.log4j.DailyRollingFileAppender
log4j.appender.error.layout=org.apache.log4j.PatternLayout
log4j.appender.error.layout.ConversionPattern=[%-5p] [%d{HH:mm:ss}] %c - %m%n
log4j.appender.error.datePattern='.'yyyy-MM-dd
log4j.appender.error.Threshold = ERROR 
log4j.appender.error.append=true
log4j.appender.error.File=${webApp.root}/WEB-INF/logs/error.log 
一、创建相关类

User类

package com.aop;
public class User {
	private String userName;	
	
	public void setUserName(String userName) {
		this.userName = userName;
	}
	public void login(){
		System.out.println("【"+this.userName+"】"+"用户正在登录。");
	}
	public void addUser(String userName,String password){
		System.out.println("【"+userName+"】"+"用户添加成功。");
	}
	public void delUser(int id){
		System.out.println("编号为【"+id+"】"+"的用户被删除。");
	}
	public void loginError(){
		throw new RuntimeException("这是特意抛出的异常信息!");
	}
}

MyAspect类

package com.aop;
import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
//日志切面
@Aspect
public class MyAspect {
	private Logger logger=Logger.getLogger(MyAspect.class);
	@Pointcut("execution(* com.aop.User.*(..))")
	//定义切入点名字
	private void allMethod(){}
	//定义前置通知
	@Before("allMethod()")
	public void before(JoinPoint joinPoint){
		//获取被调用的类名
		String targetClassName=joinPoint.getTarget().getClass().getName();
		//获取被调用的方法名
		String targetMethodName=joinPoint.getSignature().getName();
		//日志格式字符串
		String logInfoText="前置通知:"+targetClassName+"类的"+targetMethodName+"方法开始执行";
		//将日志信息写入配置的文件中
		logger.info(logInfoText);
	}
	//定义后置通知
	@AfterReturning("allMethod()")
	public void afterReturn(JoinPoint joinPoint){
		//获取被调用的类名
		String targetClassName=joinPoint.getTarget().getClass().getName();
		//获取被调用的方法名
		String targetMethodName=joinPoint.getSignature().getName();
		//日志格式字符串
		String logInfoText="后置通知:"+targetClassName+"类的"+targetMethodName+"方法开始执行";
		//将日志信息写入配置的文件中
		logger.info(logInfoText);
	}
	//定义异常通知
	@AfterThrowing(pointcut="allMethod()",throwing="e")
	public void throwing(JoinPoint joinPoint,Exception e){
		//获取被调用的类名
		String targetClassName=joinPoint.getTarget().getClass().getName();
		//获取被调用的方法名
		String targetMethodName=joinPoint.getSignature().getName();
		//日志格式字符串
		String logInfoText="异常通知:执行"+targetClassName+"类的"+targetMethodName+"方法时发生异常";
		//将日志信息写入配置的文件中
		logger.info(logInfoText);
	}
	//定义环绕通知
	@Around("allMethod()")
	public void around(ProceedingJoinPoint joinPoint)throws Throwable{
		//获取被调用的类名
		String targetClassName=joinPoint.getTarget().getClass().getName();
		//获取被调用的方法名
		String targetMethodName=joinPoint.getSignature().getName();
		//日志格式字符串
		String logInfoText="环绕通知:"+targetClassName+"类的"+targetMethodName+"方法开始执行";
		//将日志信息写入配置的文件中
		logger.info(logInfoText);
		joinPoint.proceed();
		System.out.println("环绕通知:"+targetClassName+"类的"+targetMethodName+"方法执行完毕");
	}
}

Test类

package com.aop;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
	public static void main(String[] args) {
		//加载SpringAop.xml配置
		ApplicationContext context=new ClassPathXmlApplicationContext("SpringAop.xml");
		//获取配置中的user实例
		User user=(User) context.getBean("user");
		user.login();
		user.addUser("白雪", "baixue");
		user.delUser(1);
		user.loginError();
	}
}
二、配置SpringAop.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"
    xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- bean definitions here -->
<bean id="user" class="com.aop.User">
	<property name="userName">
		<value>马向林</value>
	</property>
</bean>
<bean class="com.aop.MyAspect"></bean>
<!-- 开启基于@Aspectj切面的注解处理器 -->
<aop:aspectj-autoproxy />
</beans>

运行结果:


猜你喜欢

转载自blog.csdn.net/dwenxue/article/details/80840545