案例:有一个接口Service有一个insert方法,在insert被调用时打印调用前的毫秒数与调用后的毫秒数,其实现为
public class UserServiceImpl implements UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
public void addUser(){
System.out.println("方法开始时间:"+new Date());
userDao.addUser();
System.out.println("方法结束时间:"+new Date());
}
}
问题:输出日志的逻辑还是无法复用
AOP:全称是Aspect Oriented Programming即:面向切面编程。
简单的说它就是把我们程序重复的代码抽取出来,在需要执行的时候,使用动态代理的技术,在不修改源码的基础上,对程序进行增强:权限校验,日志记录,性能监控,事务控制.
连接点(joinpoint)
被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法。
切入点(pointcut)
切入点是指我们要对哪些连接点进行拦截的定义
通知/增强(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类
切面(aspect)
是切入点和通知的结合
引介(introduction)
是一种特殊的通知,在不修改代码的前提下,引介可以在运行期为类动态地添加一些方法或字段
目标对象(Target)
要代理的目标对象(要增强的类)
织入(weave)
将增强应用到目标的过程将advice应用到target的过程
代理(Proxy)
一个类被AOP织入增强之后,就产生一个代理类
org.springframework
spring-context
5.1.8.RELEASE
org.slf4j
slf4j-log4j12
1.7.30
org.springframework
spring-aspects
5.1.8.RELEASE
package com.by.dao;
public class UserDaoImpl implements UserDao{
@Override
public void addUser(){
System.out.println("insert into tb_user......");
}
}
package com.by.service;
import com.by.dao.UserDao;
public class UserServiceImpl implements UserService{
private UserDao userDao;
public void setUserDao(UserDao userDao){
this.userDao=userDao;
}
@Override
public void addUser(){
userDao.addUser();
// int a=10;
// System.out.println(a/0);
}
}
package com.by.web;
import com.by.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Client {
public static void main(String[] args) {
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = ac.getBean("userService", UserService.class);
userService.addUser();
System.out.println(userService.getClass());
}
}
1.创建增强类
package com.by.advice;
import org.aspectj.lang.ProceedingJoinPoint;
public class MyAdvice {
public void before(){
System.out.println("前置通知....");
}
public void after(){
System.out.println("最终通知....");
}
public void afterReturn(){
System.out.println("后置通知.....");
}
public void afterThrow(){
System.out.println("异常通知,出错了......");
}
public void around(ProceedingJoinPoint joinPoint){
try {
System.out.println("方法执行前的环绕通知.......");
joinPoint.proceed();
System.out.println("方法执行后的环绕通知........");
} catch (Throwable e) {
e.printStackTrace();
}
}
}
2.配置增强类
切点表达式
表达式语法:
execution([修饰符] 返回值类型 包名.类名.方法名(参数))
例如:
execution(* com.by.service.UserService.add(..))
execution(* com.by.service.UserService.*(..))
execution(* com.by.service.*.*(..))
配置切点
增强的类型
aop:before:用于配置前置通知
aop:after-returning:用于配置后置【try】通知,它和异常通知只能有一个执行
aop:after-throwing:用于配置异常【catch】通知,它和后置通知只能执行一个
aop:after:用于配置最终【finally】通知
aop:around:用于配置环绕通知
配置切面
org.springframework
spring-context
5.1.8.RELEASE
org.slf4j
slf4j-log4j12
1.7.30
org.springframework
spring-aspects
5.1.8.RELEASE
package by.dao;
import org.springframework.stereotype.Repository;
@Repository
public class UserDaoImpl implements UserDao{
@Override
public void addUser(){
System.out.println("insert into tb_user......");
}
}
package by.service;
import by.dao.UserDao;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class UserServiceImpl implements UserService{
@Autowired
private UserDao userDao;
@Override
public void addUser(){
userDao.addUser();
// int a=10;
// System.out.println(a/0);
}
}
package by.web;
import by.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Client {
public static void main(String[] args) {
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = ac.getBean("userServiceImpl", UserService.class);
userService.addUser();
System.out.println(userService.getClass());
}
}
常用注解
@Aspect:把当前类声明为切面类
@Before:前置通知,可以指定切入点表达式
@AfterReturning:后置【try】通知,可以指定切入点表达式
@AfterThrowing:异常【catch】通知,可以指定切入点表达式
@After:最终【finally】通知,可以指定切入点表达式
@Around:环绕通知,可以指定切入点表达式
2.注解方式实现aop
package by.advice;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
@Component
@Aspect //标识要增强的类
public class MyAdvice {
@Before("execution(* by.service.*.*(..))")
public void before(){
System.out.println("前置通知....");
}
@After("execution(* by.service.*.*(..))")
public void after(){
System.out.println("最终通知....");
}
@AfterReturning("execution(* by.service.*.*(..))")
public void afterReturn(){
System.out.println("后置通知.....");
}
@AfterThrowing("execution(* by.service.*.*(..))")
public void afterThrow(){
System.out.println("异常通知,出错了......");
}
@Around("execution(* by.service.*.*(..))")
public void around(ProceedingJoinPoint joinPoint){
try {
System.out.println("方法执行前的环绕通知.......");
joinPoint.proceed();
System.out.println("方法执行后的环绕通知........");
} catch (Throwable e) {
e.printStackTrace();
}
}
}
测试结果
1.aop的核心概念
切点(pintcut):要增强的方法,eg:add()、update()
通知/增强(advice):要搞的事情,eg:日志
前置通知:aop:before
后置通知:aop:after-returning【try】
最终通知:aop:after【finally】
异常通知:aop:after-throwing【catch】
环绕通知:aop:around
try{
...
return aop:after-returning
}catch(Exception e){
...
aop:after-throwing
}finally{
...
aop:after
}
切面(aspect):把增强应用到切点上
2、切点表达式
格式:execution([修饰符] 返回值 报名.类名.方法名(参数))
eg:execution(* com.by.service.*.*(..))
3、基于xml的aop配置
1)pom.xml
spring-context、spring-aspects、slf4j-log4j12
2)advice
public class MyLogAdvice {
public void before(){
System.out.println("前置通知....");
}
}
2)aop
4、spring基于注解的aop配置
1)pom.xml
spring-context、spring-aspects、slf4j-log4j12
2)advice
@Component
@Aspect
public class MyLogAdvice {
@Before("execution(* com.by.service.*.*(..)")
public void before(){
System.out.println("前置通知....");
}
}
5、开启spring对aop注解的支持