前往小程序,Get更优阅读体验!
立即前往
发布
社区首页 >专栏 >Spring5学习笔记AOP编程

Spring5学习笔记AOP编程

作者头像
程序员Leo
发布2023-08-16 08:31:03
发布2023-08-16 08:31:03
15100
代码可运行
举报
文章被收录于专栏:Java知识点Java知识点
运行总次数:0
代码可运行

# 1. 静态代理设计模式

# 1.1 为什么需要代理设计模式

在 JavaEE 分层开发开发中,那个层次对于我们来讲最重要

代码语言:javascript
代码运行次数:0
复制
DAO ---> Service --> Controller 

JavaEE分层开发中,最为重要的是Service层

Service 层中包含了哪些代码?

代码语言:javascript
代码运行次数:0
复制
Service层中 = 核心功能(几十行 上百代码) + 额外功能(附加功能)
1. 核心功能
   业务运算
   DAO调用
2. 额外功能(事务、日志、性能...)
   1. 不属于业务
   2. 可有可无
   3. 代码量很小 

额外功能书写在 Service 层中好不好?

Controller 层(Service 层的调用者)除了需要核心功能,还需要这些额外功能。

但是从软件设计者角度看:Service 层最好不要写额外功能。

现实生活中的解决方式

代理模式是一种比较好理解的 设计模式 。简单来说就是 我们使用代理对象来代替对真实对象 (real object) 的访问,这样就可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象的功能。

代理模式的主要作用是扩展 目标对象 的功能,比如说在目标对象的某个方法执行前后你可以增加一些自定义的操作。

代理模式:为一个对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象。这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。 被代理的对象可以是 远程对象 、创建开销大的对象或需要安全控制的对象

举个例子:当我们工作之后需要出去租房子,房东张贴广告带我看房子,最后签合同,但是房东只想坐着签合同并不想到处跑着看房子,于是就找了一个中介专门来宣传广告并且带租户看房子,而房东只负责签合同收钱!中介在这里就可以看作是代理你的 代理对象代理的行为(方法) 是带租户看房子。

# 1.2 代理设计模式分析

# 2.1 概念

通过代理类,为原始类(目标)增加额外的功能 好处:利于原始类 (目标) 的维护

# 2.2 名词解释
代码语言:javascript
代码运行次数:0
复制
1. 目标类 原始类 
   指的是 业务类 (核心功能 --> 业务运算 DAO调用)
2. 目标方法,原始方法
   目标类(原始类)中的方法 就是目标方法(原始方法)
3. 额外功能 (附加功能)
   日志,事务,性能
# 2.3 代理开发的核心要素
代码语言:javascript
代码运行次数:0
复制
代理类 = 实现和目标类相同的接口 + 在同名方法中添加额外功能 + 调用原始类同名方法

房东 ---> public interface UserService{
               m1
               m2
          }
          UserServiceImpl implements UserService{
               m1 ---> 业务运算 DAO调用
               m2 
          }
          UserServiceProxy implements UserService
               m1
               m2
# 2.4 编码

静态代理:为每一个原始类,手动编写一个代理类 (.java .class)

# 2.5 静态代理存在的问题
代码语言:javascript
代码运行次数:0
复制
1. 静态类文件数量过多,不利于项目管理
   UserServiceImpl  UserServiceProxy
   OrderServiceImpl OrderServiceProxy
2. 额外功能维护性差
   代理类中 额外功能修改复杂(麻烦)

# 2. Spring 的动态代理开发

# 2.1 Spring 动态代理的概念

概念:通过 代理类 为原始类 (目标类) 增加额外功能,代理类由 Spring 动态生成。 好处:利于 原始类 (目标类) 的维护

# 2.2 搭建开发环境

代码语言:javascript
代码运行次数:0
复制
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-aop</artifactId>
  <version>5.1.14.RELEASE</version>
</dependency>

<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjrt</artifactId>
  <version>1.8.8</version>
</dependency>

<dependency>
  <groupId>org.aspectj</groupId>
  <artifactId>aspectjweaver</artifactId>
  <version>1.8.3</version>
</dependency>

# 2.3 Spring 动态代理的开发步骤

# 1. 创建原始对象 (目标对象)
代码语言:javascript
代码运行次数:0
复制
public class UserServiceImpl implements UserService{
    @Override
    public void register(User user) {
        System.out.println("UserServiceImpl.register");
    }

    @Override
    public boolean login(String name, String password) {
        System.out.println("UserServiceImpl.login");
        return true;
    }
}
<bean id="userServiceImpl" class="com.Leo.dynamic.service.impl.UserServiceImpl"/>
# 2. 定义额外功能

实现 MethodBeforeAdvice 接口

代码语言:javascript
代码运行次数:0
复制
public class Before implements MethodBeforeAdvice {
	//作用:给原始方法添加额外功能
    //注意:会在原始方法运行之前运行此方法
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println("-----method before advice log------");
    }
}
<bean id="before" class="com.Leo.dynamic.service.Before"/>
# 3. 定义切入点
代码语言:javascript
代码运行次数:0
复制
1. 切入点:额外功能加入的位置

2. 目的:由程序员根据自己的需要,决定额外功能加入给那个原始方法
register()
login()

简单的测试:所有方法都做为切入点,都加入额外的功能。
<aop:config>
   <aop:pointcut id="pc" expression="execution(* *(..))"/>
</aop:config>
# 4. 组装
代码语言:javascript
代码运行次数:0
复制
<!-- 组装切入点与额外功能 -->
<aop:advisor advice-ref="before" pointcut-ref="pc"/>
# 5. 测试调用
代码语言:javascript
代码运行次数:0
复制
目的:获得Spring工厂创建的动态代理对象,并进行调用
ApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext.xml");
注意:
   1. Spring的工厂通过	原始对象的id值获得的是代理对象
   2. 获得代理对象后,可以通过声明接口类型,进行对象的存储
   
UserService userService=(UserService)ctx.getBean("userServiceImpl");

userService.login("","");
userService.register(new User());

控制台打印: 可以发现在日志之前输入了

# 2.4 动态代理细节分析

# 4.1 Spring 创建的动态代理类在哪里?

Spring 框架在运行时,通过动态字节码技术,在 JVM 创建的,运行在 JVM 内部,等程序结束后就消失了。 什么叫动态字节码技术:通过第三方动态字节码框架,在 JVM 中创建对应类的字节码,进而创建对象,当虚拟机结束,动态字节码跟着消失。 结论:动态代理不需要定义类文件,都是 JVM 运行过程中动态创建的,所以不会造成静态代理,类文件数量过多,影响项目管理的问题。

# 4.2 动态代理编程简化代理的开发

在额外功能不改变的前提下,创建其他目标类(原始类)的代理对象时,只需要指定原始 (目标) 对象即可。

# 3. Spring 动态代理详解

# 3.1 额外功能的详解

MethodBeforeAdvice 分析

作用: 原始方法执行之前,运行额外功能。

代码语言:javascript
代码运行次数:0
复制
public class Before implements MethodBeforeAdvice {
    /**
     * 作用:给原始方法添加额外功能
     * 注意:会在原始方法运行之前运行此方法
     * @param method 原始方法 login() register() ...
     * @param objects 原始方法的参数列表 name password ...
     * @param o 原始对象 UserServiceImpl OrderServiceImpl
     * @throws Throwable 抛出的异常
     */
    @Override
    public void before(Method method, Object[] objects, Object o) throws Throwable {
        System.out.println("---- MethodBeforeAdvice  log... ----");
    }
}

实战:需要时才用,可能都会用到,也有可能都不用。

MethodInterceptor (方法拦截器)

MethodInterceptor 接口:额外功能可定义在原始方法执行 前、后、前和后。

代码语言:javascript
代码运行次数:0
复制
public class Around implements MethodInterceptor {
    /**
     * @param invocation 封装了原始方法 invocation.proceed()表示原始方法的运行
     * @return 原始方法的返回值
     * @throws Throwable 可能抛出的异常
     */
    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        System.out.println("------ 额外功能 log -----");
		//原始方法的执行
        Object ret = invocation.proceed();
        //返回原始方法的返回值
        return ret;
    }
}

额外功能运行在原始方法执行之后

代码语言:javascript
代码运行次数:0
复制
public Object invoke(MethodInvocation invocation) throws Throwable {
  Object ret = invocation.proceed();
  System.out.println("-----额外功能运行在原始方法执行之后----");

  return ret;
}

额外功能运行在原始方法执行之前和之后(实战:事务需要在之前和之后都运行)

代码语言:javascript
代码运行次数:0
复制
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
  System.out.println("-----额外功能运行在原始方法执行之前----");
  Object ret = invocation.proceed();
  System.out.println("-----额外功能运行在原始方法执行之后----");

  return ret;
}

额外功能运行在原始方法抛出异常时

代码语言:javascript
代码运行次数:0
复制
  @Override
public Object invoke(MethodInvocation invocation) throws Throwable {
  
    Object ret = null;
    try {
      ret = invocation.proceed();
    } catch (Throwable throwable) {
      System.out.println("-----原始方法抛出异常 执行的额外功能 ---- ");
      throwable.printStackTrace();
    }
  
    return ret;
  }

MethodInterceptor 可以影响原始方法的返回值

代码语言:javascript
代码运行次数:0
复制
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
 System.out.println("------log-----");
   Object ret = invocation.proceed();
   //拿到原始方法的返回值后进行一些操作就会影响,直接返回就不影响
   return false;
}

# 3.2 切入点详解

切入点决定了额外功能加入的位置。

代码语言:javascript
代码运行次数:0
复制
<aop:pointcut id="pc" expression="execution(* *(..))"/>
exection(* *(..)) ---> 匹配了所有方法    a  b  c 

1. execution()  切入点函数
2. * *(..)      切入点表达式 
# 1. 切入点表达式

方法切入点表达式:

代码语言:javascript
代码运行次数:0
复制
*  *(..)  --> 所有方法

* ---> 修饰符 返回值
* ---> 方法名
()---> 参数表
..---> 对于参数没有要求 (0个或多个)

举例:

代码语言:javascript
代码运行次数:0
复制
# 定义login方法作为切入点
* login(..)

# 定义register作为切入点
* register(..)

# 定义名字为login且有两个字符串类型参数的方法 作为切入点
* login(String,String)

# 注意:非java.lang包中的类型,必须要写全限定名
* register(com.Leo.proxy.User)

# ..可以和具体的参数类型连用(至少有一个参数是String类型)
* login(String,..)

# 精准方法切入点限定
# 修饰符 返回值    包.类.方法(参数)

    *             com.yuziayn.proxy.UserServiceImpl.login(..)
    *             com.Leo.proxy.UserServiceImpl.login(String,String)

类切入点表达式:

指定特定的类作为切入点,即这个类中所有的方法都会加上额外功能。

举例:

代码语言:javascript
代码运行次数:0
复制
# 类中的所有方法都加入额外功能 
* com.Leo.proxy.UserServiceImpl.*(..)

# 忽略包
# 1. 类只在一级包下  com.UserServiceImpl
* *.UserServiceImpl.*(..)

# 2. 类可在多级包下  com.Leo.proxy.UserServiceImpl
* *..UserServiceImpl.*(..)

包切入点表达式:

指定包作为切入点,即这个包中的所有类及其方法都会加入额外的功能。

举例:

代码语言:javascript
代码运行次数:0
复制
# proxy包作为切入点,即proxy包下所有类中的所有方法都会加入额外功能,但是不包括其子包中的类!
* com.Leo.proxy.*.*(..)

# 当前包及其子包都生效
* com.Leo.proxy..*.*(..) 
# 2 切入点函数

作用:用于执行切入点表达式。

execution()

代码语言:javascript
代码运行次数:0
复制
最为重要的切入点函数,功能最全!
用于执行:方法切入点表达式、类切入点表达式、包切入点表达式 

弊端:execution执行切入点表达式 ,书写麻烦
     execution(* com.Leo.proxy..*.*(..))
     
注意:其他的切入点函数 只是简化execution书写复杂度,功能上完全一致

args()

代码语言:javascript
代码运行次数:0
复制
# 作用:用于函数(方法)参数的匹配

# 举例:方法参数必须得是2个字符串类型的参数
	execution(* *(String,String))
	等价于:
	args(String,String)

within()

代码语言:javascript
代码运行次数:0
复制
# 作用:用于进行类、包切入点表达式的匹配
# 举例:
# UserServiceImpl类作为切入点:
	execution(* *..UserServiceImpl.*(..))

	within(*..UserServiceImpl)
# proxy包作为切入点:
	execution(* com.Leo.proxy..*.*(..))
	
	within(com.yuziayan.proxy..*)

@annotation()

代码语言:javascript
代码运行次数:0
复制
<!-- 作用:为具有特殊注解的方法加入额外功能 -->

<aop:pointcut id="" expression="@annotation(com.baizhiedu.Log)"/>

切入点函数间的逻辑运算:

目的:整合多个切入点函数一起配合工作,进而完成更为复杂的需求。

and 与操作(同时满足)

代码语言:javascript
代码运行次数:0
复制
# 案例:方法名为login,同时有2个字符串类型的参数:
    execution(* login(String,String))

    execution(* login(..)) and args(String,String)

# 注意:与操作不能用于同种类型的切入点函数 
# 错误案例:register方法 和 login方法作为切入点(不能用and,而用or!)
    execution(* login(..)) and execution(* register(..))
# 上面的语句会发生错误,因为其实际表达的含义是方法名为login同时方法名为register,显然有悖逻辑,此时应该用到的是 or

or 或操作(满足一种即可)

代码语言:javascript
代码运行次数:0
复制
# 案例:register方法 和 login方法作为切入点 
	execution(* login(..)) or  execution(* register(..))

# 4. AOP 编程

# 4.1 AOP 概念

代码语言:javascript
代码运行次数:0
复制
# AOP (Aspect Oriented Programing)   面向切面编程 = Spring动态代理开发
# 以切面为基本单位的程序开发,通过切面间的彼此协同,相互调用,完成程序的构建
# 切面 = 切入点 + 额外功能

# OOP (Object Oriented Programing)   面向对象编程 Java
# 以对象为基本单位的程序开发,通过对象间的彼此协同,相互调用,完成程序的构建

# POP (Procedure Oriented Programing) 面向过程(方法、函数)编程 C 
# 以过程为基本单位的程序开发,通过过程间的彼此协同,相互调用,完成程序的构建

# AOP的概念:
     本质就是Spring的动态代理开发,通过代理类为原始类增加额外功能。
     好处:利于原始类的维护
# 注意:AOP编程不可能取代OOP,而是OOP编程的补充。

# 4.2 AOP 编程的开发步骤

  1. 原始对象
  2. 额外功能 (MethodInterceptor)
  3. 切入点
  4. 组装切面 (额外功能 + 切入点)

# 4.3 切面的名词解释

代码语言:javascript
代码运行次数:0
复制
切面 = 切入点 + 额外功能 

几何学
   面 = 点 + 相同的性质

# 5. AOP 的底层实现原理

# 5.1 核心问题

  • AOP 如何创建动态代理类?(动态字节码技术)
  • Spring 工厂如何加工创建代理对象?通过原始对象的 id 值,获得的是代理对象。

# 5.2 动态代理类的创建

# 1. JDK 的动态代理
  • Proxy.newProxyInstance() 方法参数详解:

编码:

代码语言:javascript
代码运行次数:0
复制
public class TestJDKProxy {
    public static void main(String[] args) {
        //1.创建原始对象
        //注意:由于后面匿名子类的方法中用到了userService,所以应该用final修饰
        //     而JDK1.8以后默认加了final,不需要手动加
        UserService userService = new UserServiceImpl();

        //2.JDK创建代理对象
        InvocationHandler handler = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("----------- JDKProxy log -----------");\
                //目标方法运行:
                Object ret = method.invoke(userService, args);
                return ret;
            }
        };

        UserService userServiceProxy = (UserService) Proxy.newProxyInstance(userService.getClass().getClassLoader(), userService.getClass().getInterfaces(),handler);

        userServiceProxy.login("Leo", "123456");
        userServiceProxy.register(new User());

    }
}
# 2. CGlib 的动态代理
  • 原理:通过父子继承关系创建代理对象。原始类作为父类,代理类作为子类,这样既可以保证 2 者方法一致,同时在代理类中提供新的实现 (额外功能 + 原始方法)

CGlib 编码:

代码语言:javascript
代码运行次数:0
复制
package com.Leo.cglib;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class TestCGlibProxy {
    public static void main(String[] args) {
        //1.创建原始对象
        UserServiceImpl userService = new UserServiceImpl();

        //2.通过CGlib创建代理对象
        //  2.1 创建Enhancer
        Enhancer enhancer = new Enhancer();
        //  2.2 设置借用类加载器
        enhancer.setClassLoader(TestCGlibProxy.class.getClassLoader());
        //  2.3 设置父类(目标类)
        enhancer.setSuperclass(userService.getClass());
        //  2.4 设置回调,额外功能写在里面
        enhancer.setCallback(new MethodInterceptor() {
            //相当于 InvocationHandler --> invoke()
            @Override
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                //额外功能:
                System.out.println("========= CGlibProxy log ========");
                //目标方法执行:
                Object ret = method.invoke(userService, objects);
                return ret;
            }
        });
        //  2.5 通过Enhancer对象创建代理
        UserServiceImpl service = (UserServiceImpl) enhancer.create();

        //测试:
        service.register();
        service.login();

    }
}
# 3. 总结
代码语言:javascript
代码运行次数:0
复制
1. JDK动态代理   Proxy.newProxyInstance()  
# 通过目标类实现的接口创建代理类 
2. Cglib动态代理 Enhancer                  
# 通过继承目标类创建代理类 

# 5.3 Spring 工厂如何返回代理对象

  • 思路分析:

编码模拟:

代码语言:javascript
代码运行次数:0
复制
public class ProxyBeanPostProcessor implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        InvocationHandler invocation = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("----------- 模拟Spring返回代理对象的方式 log -----------");

                Object ret = method.invoke(bean, args);

                return ret;
            }
        };

        return Proxy.newProxyInstance(ProxyBeanPostProcessor.class.getClassLoader(), bean.getClass().getInterfaces(), invocation);
    }
}
代码语言:javascript
代码运行次数:0
复制
<!-- 1.配置原始对象 -->
<bean id="userService" class="com.Leo.factory.UserServiceImpl"></bean>
<!-- 2.配置自己模拟的ProxyBeanPostProcessor -->
<bean id="proxyBeanPostProcessor" class="com.Leo.factory.ProxyBeanPostProcessor"/>

# 6. 基于注解的 AOP 编程

# 6.1 开发步骤:

原始对象

额外功能

切入点

组装切面

代码语言:javascript
代码运行次数:0
复制
/**
 * 声明切面类     @Aspect
 * 定义额外功能   @Around
 * 定义切入点     @Around("execution(* login(..))")
 *
 */
@Aspect
public class MyAspect {

    @Around("execution(* login(..))")//组装了切入点和额外功能
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        //额外功能:
        System.out.println("--------- 基于注解的AOP编程 log --------");
        //原始方法执行:
        Object ret = joinPoint.proceed();

        return ret;
    }
}
代码语言:javascript
代码运行次数:0
复制
<!-- 原始对象 -->
<bean id="userService" class="com.Leo.aspect.UserServiceImpl"></bean>

<!-- 切面 -->
<bean id="myAspect" class="com.Leo.aspect.MyAspect"/>

<!-- 开启基于注解的AOP编程 -->
<aop:aspectj-autoproxy/>

# 6.2 细节分析:

切入点复用:

代码语言:javascript
代码运行次数:0
复制
@Aspect
public class MyAspect {

    /**
     * 切入点复用:定义一个函数,加上@Pointcut注解,通过该注解的value定义切入点表达式,以后可以复用。
     */
    @Pointcut("execution(* login(..))")
    public void myPointcut(){}

    @Around("myPointcut()")//组装了切入点和额外功能
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        //额外功能:
        System.out.println("--------- 基于注解的AOP编程 log --------");
        //原始方法执行:
        Object ret = joinPoint.proceed();

        return ret;
    }


    @Around("myPointcut()")
    public Object around1(ProceedingJoinPoint joinPoint) throws Throwable {
        //额外功能:
        System.out.println("--------- 基于注解的AOP编程 tx --------");
        //原始方法执行:
        Object ret = joinPoint.proceed();
        return ret;
    }

}

动态代理的创建方式:

代码语言:javascript
代码运行次数:0
复制
AOP底层实现  2种代理创建方式
  1.  JDK   通过实现接口,创建代理对象
  2.  Cglib 通过继承目标类,创建代理对象
  
  默认情况 AOP编程 底层应用JDK动态代理创建方式 
  如果要切换Cglib
       1. 基于注解AOP开发
          <aop:aspectj-autoproxy proxy-target-class="true" />
       2. 传统的AOP开发
          <aop:config proxy-target-class="true">
          </aop>

# 7. AOP 开发中的一个坑

坑:在同一个业务类中,业务方法间相互调用时,只有最外层的方法,加入了额外功能 (内部的方法,通过普通的方式调用,运行的都是原始方法)。如果想让内层的方法也调用代理对象的方法,就要实现 AppicationContextAware 获得工厂,进而获得代理对象。

代码语言:javascript
代码运行次数:0
复制
public class UserServiceImpl implements UserService, ApplicationContextAware {
    private ApplicationContext ctx;


    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
              this.ctx = applicationContext;
    }

    @Log
    @Override
    public void register(User user) {
        System.out.println("UserServiceImpl.register 业务运算 + DAO ");
        //throw new RuntimeException("测试异常");

        //调用的是原始对象的login方法 ---> 核心功能
        /*
            设计目的:代理对象的login方法 --->  额外功能+核心功能
            ApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext2.xml");
            UserService userService = (UserService) ctx.getBean("userService");
            userService.login();

            Spring工厂重量级资源 一个应用中 应该只创建一个工厂
         */

        UserService userService = (UserService) ctx.getBean("userService");
        userService.login("Leo", "123456");
    }

    @Override
    public boolean login(String name, String password) {
        System.out.println("UserServiceImpl.login");
        return true;
    }
}

# 8. AOP 阶段知识总结

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-08-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • # 1. 静态代理设计模式
    • # 1.1 为什么需要代理设计模式
    • # 1.2 代理设计模式分析
      • # 2.1 概念
      • # 2.2 名词解释
      • # 2.3 代理开发的核心要素
      • # 2.4 编码
      • # 2.5 静态代理存在的问题
  • # 2. Spring 的动态代理开发
    • # 2.1 Spring 动态代理的概念
    • # 2.2 搭建开发环境
    • # 2.3 Spring 动态代理的开发步骤
      • # 1. 创建原始对象 (目标对象)
      • # 2. 定义额外功能
      • # 3. 定义切入点
      • # 4. 组装
      • # 5. 测试调用
    • # 2.4 动态代理细节分析
      • # 4.1 Spring 创建的动态代理类在哪里?
      • # 4.2 动态代理编程简化代理的开发
  • # 3. Spring 动态代理详解
    • # 3.1 额外功能的详解
    • # 3.2 切入点详解
      • # 1. 切入点表达式
      • # 2 切入点函数
  • # 4. AOP 编程
    • # 4.1 AOP 概念
    • # 4.2 AOP 编程的开发步骤
    • # 4.3 切面的名词解释
  • # 5. AOP 的底层实现原理
    • # 5.1 核心问题
    • # 5.2 动态代理类的创建
      • # 1. JDK 的动态代理
      • # 2. CGlib 的动态代理
      • # 3. 总结
    • # 5.3 Spring 工厂如何返回代理对象
  • # 6. 基于注解的 AOP 编程
    • # 6.1 开发步骤:
    • # 6.2 细节分析:
  • # 7. AOP 开发中的一个坑
  • # 8. AOP 阶段知识总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档