【Spring AOP,35岁以后的Java程序员出路在哪里

*   *   [方法切入点](about:blank#_348)

    *   [类切入点](about:blank#_390)

    *   [包切入点(实战中用的多)](about:blank#_407)

*   [切入点函数(execution、args、within)](about:blank#executionargswithin_420)

*   *   [exectuion](about:blank#exectuion_424)

    *   [args](about:blank#args_434)

    *   [within](about:blank#within_448)

    *   [@annotation](about:blank#annotation_471)

    *   [切入点函数的逻辑运算(and、or)](about:blank#andor_485)

更多内容请查看笔记目录:【Spring 5.x】学习笔记汇总

静态代理设计模式

===========================================================================

1. 为什么需要代理设计模式


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

Service 层中包含了哪些代码?

  • 核心功能(代码量较多):业务运算,DAO 调用

  • 额外功能(附加功能,不属于业务,可有可无,代码量小):事务、日志、性能 …

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

  • Service 层的调用者的角度(Controller):需要在 Service 层书写额外功能。

  • 软件设计者:Service 层不需要额外功能。

拿现实生活中的例子来做对比,解决方案是 引入一个代理

在这里插入图片描述

2. 代理设计模式


概念:通过代理类,为原始类(目标类)增加额外的功能

好处:利于原始类(目标类)的维护

名词解释

目标类 / 原始类:指的是 业务类 (核心功能 --> 业务运算、DAO调用)

目标方法 / 原始方法:目标类(原始类)中的方法就是目标方法(原始方法)

额外功能 / 附加功能:日志、事务、性能 …

代理开发的核心要素

代理类 = 目标类(原始类) + 额外功能 + 原始类(目标类)实现相同的接口


房东 --- 目标类

public interface UserService {

	m1

	m2

}

public UserServiceImpl implements UserServiceImpl {

	m1 ---> 业务运算、调用DAO

	m2 

}

----------------------------------------------------

中介 --- 代理类:要实现目标类相同的接口

public UserServiceProxy implements UserService {

	m1

	m2

} 

静态代理编码

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


public class User {} 

public interface UserService {

void register(User user);

boolean login(String name, String password);

}


public class UserServiceImpl implements UserService {

    @Override

    public void register(User user) {

        System.out.println("UserServiceImpl.register 业务运算 + DAO");

    }



    @Override

    public boolean login(String name, String password) {

        System.out.println("UserServiceImpl.login 业务运算 + DAO");

        return true;

    }

} 

/**

  • 静态代理类编码实现

*/

public class UserServiceProxy implements UserService { // 实现原始类相同的接口

private UserService userService = new UserServiceImpl(); // 代理类中必须有原始类

@Override

public void register(User user) {

    System.out.println("---log---"); // 额外功能

    userService.register(user);

}

@Override

public boolean login(String name, String password) {

    System.out.println("---log---"); // 额外功能

    return userService.login(name, password);

}

}




### [](https://gitee.com/vip204888/java-p7)静态代理存在的问题



1.  **静态类文件数量过多,不利于项目管理**  

    `UserServiceImpl`、`UserServiceProxy`  

    `OrderServiceImpl`、`OrderServiceProxy`  

    …

2.  **额外功能维护性差**:在代理类中修改额外功能较为麻烦



[](https://gitee.com/vip204888/java-p7)Spring 动态代理开发

================================================================================



概念:通过代理类为原始类(目标类)增加额外功能  

好处:利于原始类(目标类)的维护



[](https://gitee.com/vip204888/java-p7)搭建开发环境

-------------------------------------------------------------------------



org.springframework

spring-aop

5.1.14.RELEASE

org.aspectj

aspectjrt

1.8.9

org.aspectj

aspectjweaver

1.8.13




[](https://gitee.com/vip204888/java-p7)Spring 动态代理的开发步骤(5步)

---------------------------------------------------------------------------------------



1.  创建原始对象(目标对象)



public interface UserService {

void register(User user);

boolean login(String name, String password);

}


public class UserServiceImpl implements UserService {

    @Override

    public void register(User user) {

        System.out.println("UserServiceImpl.register 业务运算 + DAO");

    }



    @Override

    public boolean login(String name, String password) {

        System.out.println("UserServiceImpl.login 业务运算 + DAO");

        return true;

    }

} 

```



2.  额外功能 `MethodBeforeAdvice` 接口



```

public class Before implements MethodBeforeAdvice {

    /**

     * 作用: 把需要运行在原始方法执行之前运行的额外功能, 书写在 before 方法中

     */

    @Override

    public void before(Method method, Object[] objects, Object o) throws Throwable {

        System.out.println("---method before advice log---");

    }

} 




3.  定义 **切入点**:额外功能的加入  

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



<!--目的: 由程序员根据自己的需要,决定额外功能加入给哪个原始方法(register、login)-->
<aop:config>

    <aop:pointcut id="pc" expression="execution(* * (..))"/>

</aop:config> 



4.  组装(2、3 整合)



<beans xmlns=“http://www.springframework.org/schema/beans”

   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

   xmlns:aop="http://www.springframework.org/schema/aop"

   xsi:schemaLocation="http://www.springframework.org/schema/beans

                       http://www.springframework.org/schema/beans/spring-beans.xsd

                       http://www.springframework.org/schema/aop

                       https://www.springframework.org/schema/aop/spring-aop.xsd">



<bean id="userService" class="com.yusael.aop.UserServiceImpl"/>

<!-- 额外功能 -->

<bean id="before" class="com.yusael.aop.Before"/>



<!--切入点:额外功能的加入-->

<!--目的:由程序员根据自己的需要,决定额外功能加入给哪个原始方法(register、login)-->
<aop:config>

    <aop:pointcut id="pc" expression="execution(* * (..))"/>

    <!--表达的含义: 所有的方法 都加入before的额外功能-->

    <aop:advisor advice-ref="before" pointcut-ref="pc"/>

</aop:config>



5.  调用  

    目的:获得 Spring 工厂创建的动态代理对象,并进行调用  

    注意:

    1.  Spring 的工厂通过原始对象的 id 值获得的是代理对象

    2.  获得代理对象后,可以通过声明接口类型,进行对象的存储



/**

  • 用于测试动态代理

*/

@Test

public void test1() {

ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("/applicationContext.xml");

UserService userService = (UserService) ctx.getBean("userService");

userService.login("admin", "1234");

userService.register(new User());

}




[](https://gitee.com/vip204888/java-p7)动态代理细节分析

---------------------------------------------------------------------------



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



*   Spring 框架在运行时,通过动态字节码技术,在 JVM 创建的,运行在 JVM 内部,等程序结束后,会和 JVM 一起消失。



什么是 **动态字节码技术**?



*   通过**第三方动态字节码框架**,在 JVM 中创建对应类的字节码,进而创建对象,当虚拟机结束,动态字节码跟着消失。



结论:



*   **动态代理不需要定义类文件,都是 JVM 运行过程中动态创建的**;  

    所以不会造成静态代理的缺点:类⽂件数量过多,影响项目管理的问题。



![在这里插入图片描述](https://img-blog.csdnimg.cn/20200527211524289.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80MzczNDA5NQ==,size_16,color_FFFFFF,t_70)



* * *



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



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



* * *



动态代理使得 **额外功能的维护性大大增强**。



[](https://gitee.com/vip204888/java-p7)动态代理开发详解

===========================================================================



[](https://gitee.com/vip204888/java-p7)额外功能的详解

--------------------------------------------------------------------------



### [](https://gitee.com/vip204888/java-p7)MethodBeforeAdvice 分析



1.  `MethodBeforeAdvice` 接口作用:额外功能运行在原始方法执行之前,进行额外功能操作。



public class Before implements MethodBeforeAdvice {

/**

 * 作用: 把需要运行在原始方法执行之前运行的额外功能, 书写在 before 方法中

 *

 * Method: 额外功能所增加给的那个原始方法

 *                          login

 *                          register

 *                          --------

 *                          showOrder

 *

 * Object[]:  额外功能所增加给的那个原始方法的参数

 *                          String name,String password

 *                          User

 *                          --------

 *

 * Object: 额外功能所增加给的那个原始对象

 *                          UserServiceImpl

 *                          ---------------

 *                          OrderServiceImpl

 */

@Override

public void before(Method method, Object[] objects, Object o) throws Throwable {

    System.out.println("---new method before advice log---");

}

}




2.  `before` 方法的 3 个参数在实战中,该如何使用?  

    `before` 方法的参数,在实战中,会根据需要进行使用,不⼀定都会用到,也有可能都不用。  

    孙哥:”我用了 15 年 Spring 一次都没有用到过这个。"



### [](https://gitee.com/vip204888/java-p7)MethodInterceptor(方法拦截器)



`methodinterceptor` 接口:额外功能可以根据需要运行在原始方法执行 **前、后、前后**。



*   参数:`MethodInvocation`:额外功能所增加给的那个原始方法 (login, register)

*   返回值:`Object`:原始方法的返回值 (没有就返回 null)

*   `invocation.proceed()`:原始方法运行



额外功能运行在原始方法 **之前**:



public class Around implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

    System.out.println("---额外功能运行在原始方法执行之前---");

    Object ret = methodInvocation.proceed(); // 原始方法运行, 获取原始方法的返回值

    return ret;

}

}




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



public class Around implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

    Object ret = methodInvocation.proceed(); // 原始方法运行, 获取原始方法的返回值

    System.out.println("---额外功能运行在原始方法执行之后---");

    return ret;

}

}




额外功能运行在原始方法 **之前、之后**:



public class Around implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

	System.out.println("---额外功能运行在原始方法执行之前---");

    Object ret = methodInvocation.proceed(); // 原始方法运行, 获取原始方法的返回值

    System.out.println("---额外功能运行在原始方法执行之后---");

    return ret;

}

}




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



public class Around implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

面试题总结

面试文件获取方式:戳这里免费下载(助你面试无忧)

其它面试题(springboot、mybatis、并发、java中高级面试总结等)

terceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

	System.out.println("---额外功能运行在原始方法执行之前---");

    Object ret = methodInvocation.proceed(); // 原始方法运行, 获取原始方法的返回值

    System.out.println("---额外功能运行在原始方法执行之后---");

    return ret;

}

}




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



public class Around implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

面试题总结

面试文件获取方式:戳这里免费下载(助你面试无忧)

其它面试题(springboot、mybatis、并发、java中高级面试总结等)

[外链图片转存中…(img-3R4dfUqt-1628598989177)]

[外链图片转存中…(img-aIcALVIY-1628598989180)]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java Spring AOP (Aspect-Oriented Programming) 是 Spring 框架中的一个重要特性。它允许你通过将横切关注点(cross-cutting concerns)与程序的核心业务逻辑分离开来,以一种模块化的方式来管理和处理它们。 AOP 提供了一种在程序运行时动态地将通用功能(例如日志记录、性能计数、事务管理等)应用到应用程序中各个模块的方法上的方法。通过使用 Spring AOP,你可以将这些通用功能从核心业务逻辑中解耦出来,以便更好地维护和理解代码。 在 Spring AOP 中,你可以通过使用切点(pointcut)和通知(advice)来定义横切关注点。切点指定了在应用程序中哪些方法需要被拦截,而通知则定义了在拦截点(join point)上执行的操作。 Spring AOP 支持多种类型的通知,包括前置通知(Before advice)、后置通知(After advice)、返回通知(After-returning advice)、异常通知(After-throwing advice)和环绕通知(Around advice)。这些通知可以在方法执行前、后或周围被触发,以便执行与横切关注点相关的行为。 通过配置 Spring AOP,你可以将切点和通知与目标对象关联起来,并在运行时自动应用这些通知。这样,你就能够以一种声明式的方式管理和组织应用程序中的横切关注点,而无需修改核心业务逻辑。 总括来说,Java Spring AOPSpring 框架中的一个功能强大的特性,它通过将横切关注点与核心业务逻辑解耦,提供了一种模块化和可维护的方式来处理通用功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值