微信图片_20200505203448.jpg
不是所有的鱼儿能生活在同一片海洋
AOP
概念: 面向切面编程(Aspect Oriented Programming),利用AOP可以对业务逻辑的各个部分进行隔离。从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高开发效率。
底层原理
- 使用动态代理
- 有接口 使用JDK动态代理
JDK动态代理
创建UserDao接口实现类代理对象
使用JDK动态代理,使用Proxy类里面的方法创建代理对象
package com.aop.dao;
public interface UserDao {
public int add(int a, int b);
public String update(String id);
}
package com.aop.dao;
public class UserDaoImpl implements UserDao{
@Override
public int add(int a, int b) {
return a + b;
}
@Override
public String update(String id) {
return id;
}
}
package com.aop.dao;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class JDKProxy {
public static void main(String[] args) {
Class[] interfaces = {UserDao.class};
UserDaoImpl userDao = new UserDaoImpl();
UserDao instance = (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));
int add = instance.add(1, 2);
System.out.println(add);
}
}
// 创建代理对象
class UserDaoProxy implements InvocationHandler {
// 增强逻辑
// 把创建的是谁的代理对象,把谁传递过来
// 有参构造
private Object object;
public UserDaoProxy(Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object invoke = method.invoke(object, args);
return invoke;
}
}
AOP(术语)
连接点: 类中可被增强的方法
切入点: 实际被真正增强的方法
通知(增强): 实际增强的逻辑部分称为通知
通知有多种类型
(1)前置通知
(2)后置通知
(3)环绕通知
(4)异常通知
(5)最终通知
切面:把通知应用到切入点过程
AOP操作(准备)
Spring框架一般都是基于AspectJ实现AOP操作
AspectJ:不是Spring组成部分,独立AOP框架。一般把AspectJ和Spring框架一起使用,进行AOP操作
方式:
基于xml配置文件
基于注解方式实现(使用)
引入相关依赖

切入点表达式
作用:知道对那个类里面的那个方法进行增强
语法结构:
execution(权限修饰符 返回类型 类全路径 方法名称 参数列表)
- AspectJ注解
创建类,在类中定义方法
package com.aopanro;
public class User {
public void add() {
System.out.println("add....");
}
}
创建增强类
创建不同的方法代表不同的通知类型
package com.aopanro;
public class UserProxy {
// 前置通知
public void before() {
System.out.println("before....");
}
}
进行通知配置
在Spring配置文件中,开启注解扫描
使用注解创建User和UserProxy对象
在增强类上面添加注解@Aspect
-
在Spring配置文件中开启生成代理对象
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 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.aopanro"></context:component-scan> <!-- 开启Aspect生成代理对象 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> </beans>
```java
package com.aopanro;
import org.springframework.stereotype.Component;
@Component
public class User {
public void add() {
System.out.println("add....");
}
}
```
```java
package com.aopanro;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
@Component
@Aspect
public class UserProxy {
// 前置通知
public void before() {
System.out.println("before....");
}
}
```
配置不同类型通知
在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
```java
package com.aopanro;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
@Component
@Aspect
public class UserProxy {
@Pointcut(value = "execution(* com.aopanro.User.add(..))")
public void pointDemo() {
}
// 前置通知
@Before(value = "pointDemo()")
public void before() {
System.out.println("before....");
}
@After(value = "pointDemo()")
public void after() {
System.out.println("after....");
}
@AfterThrowing(value = "pointDemo()")
public void afterThrow() {
System.out.println("afterThrow....");
}
@AfterReturning(value = "pointDemo()")
public void afterReturning() {
System.out.println("afterReturning....");
}
@Around(value = "pointDemo()")
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{
System.out.println("around之前。。。");
proceedingJoinPoint.proceed();
System.out.println("around之后。。。");
}
}
```
@Order() 数值越小优先级越高
-
AspectJ配置文件
(1)创建两个类,一个增强类一个被增强类
(2)在Spring配置文件中创建两个类对象
package com.aopxml; public class Book { public void buy() { System.out.println("buy....."); } }
package com.aopxml; public class BookProxy { public void before() { System.out.println("before..."); } }
(3)在Spring配置文件中配置切入点
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 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"> <bean id="book" class="com.aopxml.Book"></bean> <bean id="bookProxy" class="com.aopxml.BookProxy"></bean> <!-- 配置aop增强 --> <aop:config> <!-- 配置切入点 --> <aop:pointcut id="p" expression="execution(* com.aopxml.Book.buy(..))"/> <!-- 配置切面 --> <aop:aspect ref="bookProxy"> <!-- 增强作用在具体方法上 --> <aop:before method="before" pointcut-ref="p"></aop:before> </aop:aspect> </aop:config> </beans>
完全注解方式
package com.config; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.EnableAspectJAutoProxy; @Configuration @ComponentScan(basePackages = {"com.aopanro"}) @EnableAspectJAutoProxy(proxyTargetClass = true) public class ConfigAop { }
- 没有接口 使用CGLIB动态代理
CGLIB动态代理
创建当前类子类的代理对象,增强类的方法
package com.aop.service;
public class User {
public void add() {
// ......
}
}
package com.aop.service;
public class SubUser extends User{
@Override
public void add() {
super.add();
// 增强逻辑
}
}