Spring学习:AOP概述

一、AOP概念

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

        通俗描述:不通过修改源代码方式,在主干功能里面添加新功能 。

示例: 

Spring学习:AOP概述_第1张图片

        上例是一个基本的登录功能,我们在完成其基本流程的代码之后,如果要添加一个权限判断的功能,那么传统的做法就只能修改源代码,添加if判断,判断登陆者权限。但是,采用AOP,我们可以不修改源代码,添加新的功能。我们单独编写独立的权限判断模块,并通过配置,将其配置到登录流程中。

二、AOP底层原理

        AOP底层使用了动态代理,分为两种情况的动态代理。

2.1 有接口情况

        有接口的情况,使用JDK动态代理。 

  • 创建接口实现类的代理对象,增强类的方法。

Spring学习:AOP概述_第2张图片

        如上图所示,有一个接口以及其的一个实现类,用于实现登录功能,我们的目的是不修改源代码,为登录功能添加权限验证,也就是扩展登录功能函数login()其底层的实现原理是,通过JDK动态代理,创建UserDao接口实现类的代理对象,通过代理对象来增强、扩展原功能(登录功能)。

2.2 没有接口情况

        没有接口的情况,使用CGLIB动态代理。

  • 创建子类的代理对象,增强类的方法。 

示例: 

Spring学习:AOP概述_第3张图片

        上例中,User类中有一个add方法,要在不修改源代码的前提下,对该方进行增强,由于其没有接口,所以不能采用上节创建接口实现类的代理对象的方式来增强,其采用的底层原理是创建当前类的子类的代理对象,在其子类中,对方法功能进行增强。

 2.3 JDK有接口动态代理代码示例:

         使用JDK动态代理,需要使用Proxy类里面的方法,来创建代理对象。Proxy提供了创建动态代理类和实例的静态方法,它也是由这些方法创建的所有动态代理类的超类。方法如下:

Spring学习:AOP概述_第4张图片

方法有三个参数: 

  • 第一参数,类加载器 
  • 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口 
  • 第三参数,实现这个接口 InvocationHandler,创建代理对象,写增强的部分 

JDK动态代理代码:

    (1)创建接口,定义方法

/**
 * 接口
 */
public interface UserDao {
    public int add(int a,int b);
    public String upData(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 upData(String id) {
        System.out.println("upData方法被执行...");
        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;
//            }
//        });
        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:"+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 {
        //方法之前的增强逻辑
        System.out.println("方法之前执行的增强逻辑"+method.getName()+" :传递参数..."+ Arrays.toString(args));
        //被增强的方法执行
        Object res = method.invoke(obj,args);
        //方法之后的增强逻辑
        System.out.println("方法之后的增强逻辑"+obj);
        return res;
    }
}

代码执行结果:

方法之前执行的增强逻辑add :传递参数...[1, 2]
add方法被执行...
方法之后的增强逻辑com.wyf.JDKDemo.UserDaoImpl@31cefde0
result:3

三、结束

        本文主要介绍了AOP的基本概念,以及其实现的底层基本原理。 

你可能感兴趣的:(spring学习,spring,java)