zoukankan      html  css  js  c++  java
  • SpringAOP

    AOP:面向切面编程,是Spring的两大基石之一。

    AOP:作用

    1、日志

    按照以前的写法,会造成:

    1)、代码混乱,越来越多的非业务需求(日志、验证等)加入后,原有的业务方法急剧膨胀,每个方法在处理核心业务逻辑的同时还必须兼顾其他的点。

    2)、代码分散,以日志需求为例,只是为了满足这个单一的需求,就不得不在多个模块方法中加入重复相同的日志代码,如果日志需求变化,则要大量修改。

    package aop;
    
    /**
     * @author chenpeng
     * @date 2018/6/3 21:33
     */
    public class ArithmeticCalculatorLogImpl implements ArithmeticCalculator {
        @Override
        public int add(int i, int j) {
            //每个方法都要这样加日志,麻烦
            System.out.println("add....."+(i+j));
            return i+j;
        }
    
        @Override
        public int sub(int i, int j) {
            System.out.println("sub....."+(i-j));
            return i-j;
        }
    
        @Override
        public int mul(int i, int j) {
            System.out.println("mul....."+(i*j));
            return i*j;
        }
    
        @Override
        public int div(int i, int j) {
            System.out.println("div....."+(i/j));
            return i/j;
        }
    }
    


    使用动态代理解决上述问题

    一、代理设计模式的原理:使用一个代理将对象包装起来,然后用该代理对象取代原始对象,任何对原始对象的调用都要通过代理,代理对象决定是否以及何时将方法调用转到原始对象上。    


    使用动态代理类:

    package aop;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    import java.util.ArrayList;
    import java.util.Arrays;
    
    /**
     * @author chenpeng
     * @date 2018/6/3 21:33
     */
    public class ArithmeticCalculatorLogProxyImpl {
        //要代理的对象
        private ArithmeticCalculator target;
    
        public ArithmeticCalculatorLogProxyImpl(ArithmeticCalculator target) {
            this.target = target;
        }
    
        public ArithmeticCalculator getLogProxy(){
            ArithmeticCalculator proxy=null;
    
            //代理对象由哪一个类加载器负责加载
            ClassLoader classLoader = target.getClass().getClassLoader();
            //代理方法的类型,即代理的类型中有哪些方法
            Class[] interfaces = new Class[]{ArithmeticCalculator.class};
            //当调用代理对象的方法时,应该执行的代码
            InvocationHandler i = new InvocationHandler() {
                /**
                 *
                 * @param proxy:正在返回的那个代理对象,一般情况下,在invoke方法中不使用该对象
                 * @param method:正在被调用的方法
                 * @param args:调用方法时传入的参数
                 * @return
                 * @throws Throwable
                 */
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    String methodName = method.getName();
                    //日志
                    System.out.println("The method "+methodName+" begin with "+ Arrays.asList(args));
                    //执行的方法
                    Object result = method.invoke(target,args);
                    //日志
                    System.out.println("The method "+methodName+" end with "+Arrays.asList(result));
                    return result;
                }
            };
    
            proxy = (ArithmeticCalculator) Proxy.newProxyInstance(classLoader,interfaces,i);
    
    
            return proxy;
        }
    
    }
    
    import aop.ArithmeticCalculator;
    import aop.ArithmeticCalculatorImpl;
    import aop.ArithmeticCalculatorLogProxyImpl;
    
    /**
     * @author chenpeng
     * @date 2018/6/3 21:52
     */
    public class aopTest {
        public static void main(String[] args) {
            //被代理的对象
            ArithmeticCalculator target = new ArithmeticCalculatorImpl();
            //获取代理
            ArithmeticCalculator proxy = new ArithmeticCalculatorLogProxyImpl(target).getLogProxy();
    
            int result = proxy.add(1,2);
            System.out.println(result);
        }
    
    }
    


    虽然动态代理可以解决这些麻烦的问题,但是还不够强大,AOP更加强大

    AOP

    面向切面编程,是一种新的方法论,是对传统的OOP(面向对象编程)的补充。

    AOP的主要编程对象是切面,而切面模块化横切关注点。

    在应用AOP编程的时候,仍然需要定义公共功能,但可以明确的定义这个功能在哪里,以什么方式应用,并且不必修改受影响的类,这样一来,横切关注点就被模块化到特殊的对象(切面)里

    AOP的好处:

    —每个事物逻辑位于一个位置,代码不分散,便于维护升级

    —业务模块更加简洁,只包含核心的业务代码。


    切面:横切关注点抽取出被模块化的特殊对象。

    通知:切面必须要完成的工作。(就是切面中具体的方法,如验证、日志)

    目标:被通知的对象(也就是上图的业务逻辑)

    代理:向目标对象应用通知之后创建的对象

    连接点:程序执行的某个特定的位置,如类的某个方法调用前、调用后、方法抛出异常后等。连接点由两个信息确定:方法表示程序执行点,相对点表示的方位

    切点:每个类拥有多个连接点,AOP通过切点定位到特定的连接点。类比:连接点相当于数据库中的记录,切点相当于查询条件。切点和连接点不是一一对应的,一个切点对应多个连接点,切点通过Pointcut接口进行描述。使用类和方法作为连接点的查询条件。


  • 相关阅读:
    git错误操作导致代码丢失找回
    org.apache.ibatis.binding.BindingException: Invalid bound statement (not found)问题
    WebMvcConfigurerAdapter在Spring boot2.x已废弃
    AJAX实现模拟注册跳转
    遍历字符串替换实例
    spring boot2.x依赖
    Thymeleaf页面添加th:value后报错原因
    thymeleaf中th:href字符拼接
    刁肥宅数据结构课设“布隆过滤器的实现和应用”源代码(v1.0,永不上交)
    数据结构实验1:C++实现静态顺序表类
  • 原文地址:https://www.cnblogs.com/huangzhe1515023110/p/9276056.html
Copyright © 2011-2022 走看看