• SpringAOP的概述与实现


    目录

    SpringAOP的概述

    什么是AOP

    AOP能干什么

    AOP的特点

    AOP底层实现

    AOP基本概念

    连接点

    切入点

    通知

    切面

    目标对象

    织入

    引入

    谈谈你对AOP的理解?

    SpringAOP的实现

    依赖引用

    spring.xml配置

    注解实现

    1.定义切面 设置通知

    2.开启aop

    3.测试

    xml实现

    1.写切面类

    2.配置aop相关配置

    3.测试


    SpringAOP的概述

    什么是AOP

            Aspect oriented Programing面向切面编程,相比较oop 面向对象编程来说,Aop关注的不再是程序代码中某个类,某些方法,而aop考虑的更多的是一种面到面的切入,即层与层之间的一种切入,所以称之为切面。如:大家吃的汉堡(中间夹肉)。

    AOP能干什么

            AOP主要应用于日志记录,性能统计,安全控制,事务处理等方面,实现公共功能性的重复使用。

    AOP的特点

    1. 降低模块与模块之间的耦合度,提高业务代码的聚合度(高内聚低耦合)
    2. 提高了代码的复用性
    3. 提高系统的扩展性。(高版本兼容低版本)
    4. 可以在不影响原有的功能基础上添加新的功能

    AOP底层实现

            动态代理(JDK与CGLIB)

    AOP基本概念

    连接点

            被拦截到的每个点,spring中指被拦截到的每一个方法,spring aop一个连接点即代表一个方法的执行。

    切入点

            对连接点进行拦截的定义(匹配规则定义规定拦截哪些方法,对哪些方法进行处理),spring有专门的表达式语言定义。
     

    通知

            拦截到每一个连接点即(每一个方法)后所要做的操作

    1. 前置通知(前置增强)- before()执行方法前通知
    2. 返回通知(返回增强)- afterReturn方法正常结束返回后的通知
    3. 异常抛出通知(异常抛出增强)- afetrThrow()
    4. 最终通知- after无论方法是否发生异常,均会执行该通知。
    5. 环绕通知– around包围一个连接点(join point)的通知,如方法调用。这是最强大的一种通知类型。环绕通知可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它们自己的返回值或抛出异常来结束执行。

    切面

            切入点与通知的结合,决定了切面的定义,切入点定义了要拦截哪些类的哪些方法,通知则定义了拦截过方法后要做什么,切面则是横切关注点的抽象,与类相似,类是对物体特征的抽象,切面则是横切关注点抽象。

    目标对象

            被代理的对象

    织入

            将切面应用到目标对象并生成代理对象的这个过程即为织入

    引入

            在不修改原有应用程序代码的情况下,在程序运行期为类动态添加方法或者字段的过程称为引入

    谈谈你对AOP的理解?

    1. Aop是面向切面编程,Aop关注的不再是程序代码中某个类,某些方法,而是层与层之间的一种切入
    2. 应用于日志记录,性能统计,安全控制,事务处理等方面,实现公共功能性的重复使用
    3. 作用:降低模块与模块之间的耦合度,提高业务代码的聚合度,提高了代码的复用性,可以在不影响原有的功能基础上添加新的功能
    4.   动态代理(JDK与CGLIB)实现AOP

    SpringAOP的实现

    依赖引用

    1. org.aspectj
    2. aspectjweaver
    3. 1.9.6

    spring.xml配置

    1. xmlns:aop="http://www.springframework.org/schema/aop"
    2. http://www.springframework.org/schema/aop
    3. http://www.springframework.org/schema/aop/spring-aop.xsd

    注解实现

    1.定义切面 设置通知

    1. package com.lsf.aspect;
    2. import org.aspectj.lang.annotation.*;
    3. import org.springframework.stereotype.Component;
    4. /**
    5. * 定义切面
    6. * 定义 切入点 与 通知 的结合
    7. *
    8. */
    9. @Component //将该类交给IOC实例化
    10. @Aspect //声明此类为切面类
    11. public class LogCut {
    12. /**
    13. * 定义切入点
    14. * 通过Pointcut定义规则
    15. * @Pointcut("execution(* com.lsf.service.*.*(..))") com.lsf.service下所有方法
    16. * 例:
    17. * 1,拦截所有方法
    18. * @Pointcut("execution(* *(..))")
    19. *
    20. * 2,拦截所有公共的set方法方法
    21. * @Pointcut("execution(public set*(..))")
    22. *
    23. *
    24. * *
    25. * *
    26. */
    27. @Pointcut("execution(* com.lsf.service.*.*(..))")
    28. public void cut(){
    29. }
    30. /**
    31. * 前置通知,引用在切入点
    32. * 在目标方法调用前
    33. */
    34. @Before(value = "cut()")
    35. public void before(){
    36. System.out.println("这是一个前面通知");
    37. }
    38. /**
    39. * 返回通知,引用在切入点
    40. * 在目标方法无异常返回时输出
    41. */
    42. @AfterReturning(value = "cut()")
    43. public void afterReturn(){
    44. System.out.println("返回通知");
    45. }
    46. /**
    47. * 最终通知,引用在切入点
    48. * 在目标方法是否异常 ,都输出
    49. */
    50. @After(value = "cut()")
    51. public void after(){
    52. System.out.println("最终通知");
    53. }
    54. /**
    55. * 异常通知,引用在切入点
    56. * 在目标方法发生异常时 ,都输出
    57. */
    58. @AfterThrowing(value = "cut()",throwing = "e")
    59. public void afterThrow(Exception e){
    60. System.out.println("异常通知:异常原因: ");
    61. }
    62. /**
    63. * 环绕通知,引用在切入点
    64. * 在目标方法发生异常时 ,都输出
    65. */
    66. @Around(value = "cut()")
    67. public Object around(){
    68. System.out.println("前置通知");
    69. Object result = null;
    70. try {
    71. System.out.println("返回通知");
    72. }catch (Exception e){
    73. e.printStackTrace();
    74. System.out.println("异常通知");
    75. }catch (Throwable throwable){
    76. throwable.printStackTrace();
    77. }finally {
    78. System.out.println("最终通知");
    79. }
    80. return result;
    81. }
    82. }

    2.开启aop

      

    3.测试

    1. package com.lsf;
    2. import com.lsf.service.UserService;
    3. import org.springframework.context.ApplicationContext;
    4. import org.springframework.context.support.ClassPathXmlApplicationContext;
    5. public class Starter {
    6. public static void main(String[] args) {
    7. ApplicationContext ac = new ClassPathXmlApplicationContext("spring.xml");
    8. UserService userService = (UserService) ac.getBean("userService");
    9. userService.test();
    10. }
    11. }

     

    xml实现

    1.写切面类

    1. package com.lsf.aspect;
    2. import org.aspectj.lang.annotation.*;
    3. import org.springframework.stereotype.Component;
    4. /**
    5. * 定义切面
    6. * 定义 切入点 与 通知 的结合
    7. *
    8. */
    9. @Component //将该类交给IOC实例化
    10. @Aspect //声明此类为切面类
    11. public class LogCutxml {
    12. /**
    13. * 定义切入点
    14. */
    15. public void cut(){
    16. }
    17. /**
    18. * 前置通知,引用在切入点
    19. * 在目标方法调用前
    20. */
    21. public void before(){
    22. System.out.println("这是一个前面通知");
    23. }
    24. /**
    25. * 返回通知,引用在切入点
    26. * 在目标方法无异常返回时输出
    27. */
    28. public void afterReturn(){
    29. System.out.println("返回通知");
    30. }
    31. /**
    32. * 最终通知,引用在切入点
    33. * 在目标方法是否异常 ,都输出
    34. */
    35. public void after(){
    36. System.out.println("最终通知");
    37. }
    38. /**
    39. * 异常通知,引用在切入点
    40. * 在目标方法发生异常时 ,都输出
    41. */
    42. public void afterThrow(Exception e){
    43. System.out.println("异常通知:异常原因: ");
    44. }
    45. /**
    46. * 环绕通知,引用在切入点
    47. * 在目标方法发生异常时 ,都输出
    48. */
    49. public Object around(){
    50. System.out.println("前置通知");
    51. Object result = null;
    52. try {
    53. System.out.println("返回通知");
    54. }catch (Exception e){
    55. e.printStackTrace();
    56. System.out.println("异常通知");
    57. }catch (Throwable throwable){
    58. throwable.printStackTrace();
    59. }finally {
    60. System.out.println("最终通知");
    61. }
    62. return result;
    63. }
    64. }

    2.配置aop相关配置

    1. "logCutxml">
    2. "cut" expression="execution(* com.lsf.service..*.*(..))"/>
    3. "before" pointcut-ref="cut" />
    4. "afterReturn" pointcut-ref="cut"/>
    5. "afterThrow" throwing="e" pointcut-ref="cut" />
    6. "after" pointcut-ref="cut" />
    7. "around" pointcut-ref="cut"/>

    3.测试

    1. package com.lsf;
    2. import com.lsf.service.UserService;
    3. import org.springframework.context.ApplicationContext;
    4. import org.springframework.context.support.ClassPathXmlApplicationContext;
    5. public class Starter {
    6. public static void main(String[] args) {
    7. ApplicationContext ac = new ClassPathXmlApplicationContext("spring.xml");
    8. UserService userService = (UserService) ac.getBean("userService");
    9. userService.test();
    10. }
    11. }

     

  • 相关阅读:
    在单片机中什么是FLASH
    鸿蒙布局List简介
    Day31 Web端自动化基础
    瀑布式开发和敏捷开发
    IntelliJ IDEA 常用的插件
    三分钟搞懂MySQL5.6优化&索引下推
    一起Talk Android吧(第三百九十回:关于Android版本12适配蓝牙权限的问题)
    JavaBean处理器之MapStruct
    ChatGPT高级数据分析功能
    C Primer Plus(6) 中文版 第6章 C控制语句:循环 6.2 while语句
  • 原文地址:https://blog.csdn.net/weixin_47514459/article/details/126909391