一:AOP(概念)
1、什么是 AOP
(1)面向切面编程(方面),利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
(2)通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
二:AOP(底层原理)
AOP(JDK 动态代理)
1、使用 JDK 动态代理,使用 Proxy 类里面的方法创建代理对象
(1)调用 newProxyInstance 方法
方法有三个参数:
第一参数,类加载器
第二参数,增强方法所在的类,这个类实现的接口,支持多个接口
第三参数,实现这个接口 InvocationHandler,创建代理对象,写增强的部分
2、编写 JDK 动态代理代码
(1)创建接口,定义方法
public interface UserDAO { public int add(int a,int b);//return a+b public String update(String id); }
(2)创建接口实现类,实现方法
public class UserDAOImpl implements UserDAO{ @Override public int add(int a, int b) { System.out.println("add方法执行了......"); return a+b; } @Override public String update(String id) { System.out.println("update方法执行了......"); return id; } }
(3)使用 Proxy 类创建接口代理对象
public class JDKproxy { public static void main(String[] args) { //创建接口实现类代理对象 Class[] interfaces={UserDAO.class}; /*Proxy.newProxyInstance(JDKproxy.class.getClassLoader(), interfaces, new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { return null; } });//匿名内部类*/ /* 第一参数,类加载器 第二参数,增强方法所在的类,这个类实现的接口, 支持多个接口 第三参数,实现这个接口 InvocationHandler, 创建代理对象,写增强的部分 */ UserDAOImpl userDao = new UserDAOImpl(); UserDAO dao = (UserDAO)Proxy.newProxyInstance(JDKproxy.class.getClassLoader(), interfaces, new UserDAOProxy(userDao)); int result = dao.add(1, 2); System.out.println("结果="+result); } } //创建代理对象代码 class UserDAOProxy implements InvocationHandler{ //1把创建的是谁的代理对象,把谁传递过来 //有产构传递 private Object obj; public UserDAOProxy(Object obj){ this.obj=obj; } //增强的逻辑 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //proxy 代理对象 method表示当前的方法 args参数 System.out.println("...方法之前...."+method.getName()+".....传递参数..."+ Arrays.toString(args));//当前执行的是哪个方法 //执行当前的方法 Object res = method.invoke(obj, args);//1.对象 2.参数 //方法之后 System.out.println("...方法之后执行..."+obj); return res; } }
效果图
三:AOP(术语)
3.1几种通知类型
//前置通知@Before()
//最终通知@After() :在方法之后执行
//后置通知(返回通知)@AfterReturning();在方法返回值之后执行
//异常通知@AfterThrowing()
/环绕通知@Around//方法之前和之后都执行
四:AOP 操作(准备工作)
1、Spring 框架一般都是基于 AspectJ 实现 AOP 操作
(1)AspectJ 不是 Spring 组成部分,独立 AOP 框架,一般把 AspectJ 和 Spirng 框架一起使用,进行 AOP 操作
2、基于 AspectJ 实现 AOP 操作
(1)基于 xml 配置文件实现
(2)基于注解方式实现(使用)
3、在项目工程里面引入 AOP 相关依赖
4、切入点表达式在通知类型()中写
(1)切入点表达式作用:知道对哪个类里面的哪个方法进行增强
(2)语法结构: execution([权限修饰符] [返回类型] [类全路径] 方法名称 )
//*表示通配符 返回类型可以省略
execution(* com.xbfinal.spring.day02.User.add()) //*表示通配符 返回类型可以省略
举例 3:对 com.atguigu.dao 包里面所有类,类里面所有方法进行增强
execution(* com.xbfianl.dao.. (…))
五:AOP 操作(AspectJ 注解)
5.1、创建类,在类里面定义方法
//被增强的类 @Component public class User { public void add(){ /* int i=10/0;*/ System.out.println("add......"); } }
5. 2、创建增强类(编写增强逻辑)
(1)在增强类里面,创建方法,让不同方法代表不同通知类型
(2)使用注解创建 User 和 UserProxy 对象
(3)在增强类上面添加注解 @Aspect
(4)在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
(5)相同的切入点抽取
package com.xbfinal.spring.day02; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.*; import org.springframework.stereotype.Component; //增强类 @Component @Aspect//生成代理对象 public class UserProxy { //相同切入点抽取 @Pointcut(value = "execution(* com.xbfinal.spring.day02.User.add())") public void pointcut(){} //在增强类里面创建方法,不同的方法代表不同的通知类型 //前置通知@Before() @Before(value="pointcut()") public void before(){ System.out.println("前置通知before......"); } //最终通知:在方法之后执行 @After(value="pointcut()") public void After(){ System.out.println("最终通知After......"); } //后置通知(返回通知);在方法返回值之后执行 @AfterReturning(value="pointcut()") public void AfterReturning(){ System.out.println("后置通知AfterReturning......"); } //异常通知 @AfterThrowing(value="pointcut()") public void AfterThrowing(){ System.out.println("异常通知AfterThrowing......"); } //环绕通知 //方法之前和之后都执行 @Around(value="pointcut()") public void Around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { System.out.println("环绕通知的方法之前......"); //被增强的方法 proceedingJoinPoint.proceed(); System.out.println("环绕通知的后都执行"); } }
5.3、进行通知的配置
(1)在 spring 配置文件中,开启注解扫描
(2)在 spring 配置文件中开启生成代理对象
<?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:context="http://www.springframework.org/schema/context" 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/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!--开启注解扫描--> <context:component-scan base-package="com.xbfinal.spring.day02"></context:component-scan> <!--开启Aspect生产代理对象--> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> <!--扫描是否有@Aspect有就生产代理对象--> </beans>
5.4其他细节
一:多个增强类多同一个方法进行增强,设置增强类优先级
(1)在增强类上面添加注解 @Order(数字类型值),数字类型值越小优先级越高
@Component
@Aspect
@Order(1)
public class PersonProxy
二:完全使用注解开发 (1)创建配置类,不需要创建 xml 配置文件
@Configuration
@ComponentScan(basePackages = {“com.atguigu”})
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class ConfigAop {
}
对上面5.2代码的测试实列
public class Testaop { @Test public void testAop(){ ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml"); User user = context.getBean("user", User.class); user.add(); } }