Spring---AOP

微信图片_20200505203448.jpg
                                                   不是所有的鱼儿能生活在同一片海洋

AOP

概念: 面向切面编程(Aspect Oriented Programming),利用AOP可以对业务逻辑的各个部分进行隔离。从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高开发效率。

底层原理

  1. 使用动态代理
  • 有接口 使用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配置文件

        基于注解方式实现(使用)

    引入相关依赖

        ![image-20210404193509709.png](https://upload-images.jianshu.io/upload_images/19868552-b00e34ca4c7d03cc.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

切入点表达式

作用:知道对那个类里面的那个方法进行增强

语法结构:

     execution(权限修饰符  返回类型  类全路径  方法名称  参数列表)
  1. 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() 数值越小优先级越高
  1. 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();
        // 增强逻辑
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容