springAop 环绕通知MethodInterceptor实现对结果进行缓存的计算器

Aop原理

SpringAop 原理就是动态代理

对于实现接口的目标类使用的是jdk动态代理

对于没有实现任何接口的目标类,使用的是cglib的动态代理

代理类是程序在运行期间由JVM根据反射等机制动态生成的自动生成代理类和代理对象。

所谓动态就是指在程序运行前不存在代理类的字节码文件。

SpringAop的配置方式

三种配置方式

一:SpringAop1.x 使用ProxyFactoryBean手动配置

二:SpringAop2.x 基于命名控件的配置

三:Annotation 基于注解的配置(推荐)

Advice类型

SpringAop支持五种类型的通知(增强)
在这里插入图片描述
注意:多个Advice之间不允许有耦合,即多个Advice之间不允许有业务交叉。

(1):SpringAop1.x 使用ProxyFactoryBean 手动代理

配置方式:

基本用法: 添加jar包

 <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.11</version>
            <scope>test</scope>
        </dependency>
        <!--ioc01-core-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
        </dependency>
        <!--ioc01-bean-->
        <!-- https://mvnrepository.com/artifact/org.springframework/spring-beans -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-beans</artifactId>
        </dependency>
        <!--ioc01-context-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
        </dependency>
        <!--ioc01-expression-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-expression</artifactId>
        </dependency>
        <!--Aop依赖-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aspects</artifactId>
        </dependency>
        <!--cglib技术-->
        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
        </dependency>

1:配置Advice通知

1定义增强类,实现相应的接口

package springaop06.advice;

import org.aopalliance.intercept.Joinpoint;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;

/**
 * package_name:springaop06.advice
 *
 * @author:徐亚远 Date:2020/2/20 13:22
 * 项目名:springDemo01
 * Description:TODO
 * Version: 1.0
 **/

public class CacheAdvice implements MethodInterceptor {
    //定义一个缓存
    Map<Key, Object> cache = new HashMap<>();

    /**
     * Implement this method to perform extra treatments before and
     * after the invocation. Polite implementations would certainly
     * like to invoke {@link Joinpoint#proceed()}.
     *
     * @param invocation the method invocation joinpoint
     * @return the result of the call to {@link Joinpoint#proceed()};
     * might be intercepted by the interceptor
     * @throws Throwable if the interceptors or the target object
     *                   throws an exception
     */
    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        Method method = invocation.getMethod();
        Object[] args = invocation.getArguments();
        Object target = invocation.getThis();
        Key key = new Key();
        key.setMethod(method);
        key.setArgs(args);
        key.setTarget(target);
        //判断是否执行过这个方法
        if (cache.containsKey(key)) {
            //如果执行过直接返回值
            return cache.get(key);
        }
        System.out.println("methodName:"+method.getName()+" "+"args:"+Arrays.toString(args)+" "+"target"+target);
        Object proceed = invocation.proceed();
        //如果没有执行过这个方法把它加入到缓存中将结果加入到缓存中
        cache.put(key, proceed);
        return proceed;
    }
}

自定义Key

package springaop06.advice;

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Objects;

/**
 * package_name:springaop06.key
 *
 * @author:徐亚远 Date:2020/2/20 13:42
 * 项目名:springDemo01
 * Description:自定义Key,判断方法是否被执行过的三要素:method args target
 * Version: 1.0
 **/

public class Key {
    private Method method;
    private Object[] args;
    private Object target;

    //判断Key的唯一性重写hashCode()和equals()方法
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Key key = (Key) o;
        return Objects.equals(method, key.method) &&
                Arrays.equals(args, key.args) &&
                Objects.equals(target, key.target);
    }

    @Override
    public int hashCode() {

        int result = Objects.hash(method, target);
        result = 31 * result + Arrays.hashCode(args);
        return result;
    }

    public Method getMethod() {
        return method;
    }

    public void setMethod(Method method) {
        this.method = method;
    }

    public Object[] getArgs() {
        return args;
    }

    public void setArgs(Object[] args) {
        this.args = args;
    }

    public Object getTarget() {
        return target;
    }

    public void setTarget(Object target) {
        this.target = target;
    }
}

对结果计算的接口

package springaop06.service;


/**
 * package_name:springaop06.service
 *
 * @author:徐亚远 Date:2020/2/20 13:26
 * 项目名:springDemo01
 * Description:TODO
 * Version: 1.0
 **/
public interface CacheService {
    int add(int a,int b);
    double del(int a,int b);
}

对结果进行计算的实现类

package springaop06.service.impl;

import springaop06.service.CacheService;

/**
 * package_name:springaop06.service.impl
 *
 * @author:徐亚远 Date:2020/2/20 13:27
 * 项目名:springDemo01
 * Description:TODO
 * Version: 1.0
 **/

public class CacheServiceImpl implements CacheService {
    @Override
    public int add(int a, int b) {
        System.out.println("cacheServiceImpl  add();");
        return a+b;
    }

    @Override
    public double del(int a, int b) {
        System.out.println("cacheServiceImpl  del()");
        return a-b;
    }
}

配置spring.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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--springaop 1.x配置-->
    <!--目标类实例-->
    <bean id="cacheServiceTarget" class="springaop06.service.impl.CacheServiceImpl"/>

    <!--配置通知-->
    <bean id="cacheAdvice" class="springaop06.advice.CacheAdvice"/>

    <!--配置advisor-->
    <bean id="cacheAdvisor" class="org.springframework.aop.support.NameMatchMethodPointcutAdvisor">
        <!--配置通知-->
        <property name="advice" ref="cacheAdvice"/>
        <!--配置切点-->
        <property name="mappedNames">
            <list>
                <value>add</value>
                <value>del</value>
            </list>
        </property>
    </bean>
    <!--配置代理-->
    <bean id="cacheService" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!--配置目标类实例-->
        <property name="target" ref="cacheServiceTarget"/>
        <!--配置目标实例接口列表-->
        <property name="interfaces">
            <list>
                <value>springaop06.service.CacheService</value>
            </list>
        </property>
        <!--配置交叉业务-->
        <property name="interceptorNames">
            <list>
                <value>cacheAdvisor</value>
            </list>
        </property>
    </bean>
</beans>

书写测试类

package springaop06.controller;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import springaop06.service.CacheService;

/**
 * package_name:springaop06.controller
 *
 * @author:徐亚远 Date:2020/2/20 13:23
 * 项目名:springDemo01
 * Description:TODO
 * Version: 1.0
 **/

public class CacheTest {
    public static void main(String [] args){
        ApplicationContext ac = new ClassPathXmlApplicationContext("aop06/spring.xml");
      CacheService cacheService = (CacheService) ac.getBean("cacheService");
        System.out.println(cacheService.add(2,4 ));
        System.out.println(cacheService.add(2,4 ));
        System.out.println(cacheService.add(4,2 ));
        System.out.println(cacheService.add(4,4 ));
        System.out.println("=========================");
        System.out.println(cacheService.del(4,5 ));
        System.out.println(cacheService.del(4,5 ));
        System.out.println(cacheService.del(5,5 ));
        System.out.println(cacheService.del(5,6 ));
      System.out.println(cacheService.getClass());
    }
}

测试结果如图:
在这里插入图片描述

发布了64 篇原创文章 · 获赞 1 · 访问量 885

猜你喜欢

转载自blog.csdn.net/weixin_43311650/article/details/104426448