Day381.AOP编程 -Spring5

Posted 阿昌喜欢吃黄桃

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Day381.AOP编程 -Spring5相关的知识,希望对你有一定的参考价值。

AOP编程

⼀、静态代理设计模式

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

1.1 问题

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

    • DAO —> Service --> Controller
    • JavaEE分层开发中,最为重要的是Service层
  • Service层中包含了哪些代码?

    • Service层中 = 核⼼功能(⼏⼗⾏ 上百代码) + 额外功能(附加功能)

      1. 核⼼功能

        业务运算
        DAO调⽤

      2. 额外功能

        不属于业务

        可有可⽆

        代码量很⼩

      事务、⽇志、性能…

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

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

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

  • 现实⽣活中的解决⽅式


2. 代理设计模式

1.1 概念

通过代理类,为原始类(⽬标)增加额外的功能

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


1.2名词解释

  1. ⽬标类 原始类被代理类
    指的是 业务类 (核⼼功能 --> 业务运算 DAO调⽤)
  2. ⽬标⽅法,原始⽅法
    ⽬标类(原始类)中的⽅法 就是⽬标⽅法(原始⽅法)
  3. 额外功能 (附加功能)
    ⽇志,事务,性能

1.3 代理开发的核⼼要素

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

房东 ---> public interface UserService{
            m1
            m2
}

UserServiceImpl implements UserService{
            m1 ---> 业务运算 DAO调⽤
            m2
}

UserServiceProxy implements UserService{
            m1
            m2
}

1.4 编码

静态代理为每⼀个原始类,⼿⼯编写⼀个代理类 (.java .class)


1.5 静态代理存在的问题

  1. 静态类⽂件数量过多,不利于项⽬管理
    UserServiceImpl —> UserServiceProxy
    OrderServiceImpl —> OrderServiceProxy
  2. 额外功能维护性差
    代理类中 额外功能 修改复杂(麻烦)

⼆、Spring的动态代理开发

1. Spring动态代理的概念

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


2. 搭建开发环境

<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>

3. Spring动态代理的开发步骤

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

    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");
            return true;
        }
    }
    
    <bean id="userService" class="com.baizhiedu.proxy.UserServiceImpl"/>
    
  2. 额外功能 MethodBeforeAdvice接⼝

    额外的功能书写在接⼝的实现中,运⾏在原始⽅法执⾏之前运⾏额外功能。

    public class Before implements MethodBeforeAdvice {
        /*
    作⽤:需要把运⾏在原始⽅法执⾏之前运⾏的额外功能,书写在before⽅法中
    */
        @Override
        public void before(Method method, Object[] args, Object target) throws Throwable {
            System.out.println("-----method before advice log------");
        }
    }
    
    <bean id="before" class="com.baizhiedu.dynamic.Before"/> 
    
  3. 定义切⼊点

    切⼊点额外功能加⼊的位置
    ⽬的:由程序员根据⾃⼰的需要,决定额外功能加⼊给那个原始⽅法
    register
    login
    简单的测试所有⽅法都做为切⼊点,都加⼊额外的功能。

    <aop:config>
        <aop:pointcut id="pc" expression="execution(* *(..))"/>
    </aop:config>
    
  4. 组装 (2/3步整合)

    表达的含义:所有的⽅法 都加⼊ before的额外功能

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

  5. 调⽤

    ⽬的:获得Spring⼯⼚创建的动态代理对象,并进⾏调⽤

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

    注意:

    1. Spring的⼯⼚通过原始对象的id值获得的是代理对象
    2. 获得代理对象后,可以通过声明接⼝类型,进⾏对象的存储
    UserService userService=(UserService)ctx.getBean("userService");
    userService.login("");
    userService.register();
    


4.动态代理细节分析

  1. Spring创建的动态代理类在哪⾥?

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

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

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

  2. 动态代理额外功能的维护性⼤⼤增强


三、Spring动态代理详解

1. 额外功能的详解

  • MethodBeforeAdvice分析

    1. MethodBeforeAdvice接⼝作⽤:额外功能运⾏在原始⽅法执⾏之前,进⾏额外功能操作。

      public class Before1 implements MethodBeforeAdvice {
          /*
      作⽤:需要把运⾏在原始⽅法执⾏之前运⾏的额外功能,书写在before⽅法中
      
      方法参数如下:↓↓↓↓↓↓↓
      Method: 额外功能所增加给的那个原始⽅法
              login⽅法
              register⽅法
              showOrder⽅法
      Object[]: 额外功能所增加给的那个原始⽅法的参数。
      		String name,String password
      		User user
      
      Object: 额外功能所增加给的那个原始对象 UserServiceImpl OrderServiceImpl
      */
          @Override
          public void before(Method method, Object[] args, Object target) throws Throwable {
              System.out.println("-----new method before advice log------");
          }
      
      }
      

    2. before⽅法的3个参数在实战中,该如何使⽤。
      before⽅法的参数,在实战中,会根据需要进⾏使⽤,不⼀定都会⽤到,也有可能都不⽤

      Servlet{
          service(HttpRequest request,HttpResponse response){
              request.getParameter("name") -->
              response.getWriter() --->
          }
      }
      
  • MethodInterceptor(⽅法拦截器)推荐

    methodinterceptor接⼝:额外功能可以根据需要运⾏在原始⽅法执⾏ 前、后、前后

    public class Arround implements MethodInterceptor {
        /*
    invoke⽅法的作⽤:额外功能书写在invoke
        额外功能 
        原始⽅法之前
        原始⽅法之后
        原始⽅法执⾏之前 之后
        确定:原始⽅法怎么运⾏
    参数:MethodInvocation (Method):额外功能所增加给的那个原始⽅法
        login
        register
        invocation.proceed() ---> login运⾏
        register运⾏
        
    返回值:Object: 原始⽅法的返回值
    Date convert(String name)
    */
        @Override
        public Object invoke(MethodInvocation invocation) throws Throwable {
            System.out.println("-----额外功能 log----");
            Object ret = invocation.proceed();
            return ret;
        }
    }
    

    额外功能运⾏在原始⽅法执⾏之后

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable{
        Object ret = invocation.proceed();
        System.out.println("-----额外功能运⾏在原始⽅法执⾏之后----");
        return ret;
    }
    

    额外功能运⾏在原始⽅法执⾏之前,之后

    什么样的额外功能 运⾏在原始⽅法执⾏之前,之后都要添加?事务

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable{
        System.out.println("-----额外功能运⾏在原始⽅法执⾏之前----");
        Object ret = invocation.proceed();
        System.out.println("-----额外功能运⾏在原始⽅法执⾏之后----");
        return ret;
    }
    

    额外功能运⾏在原始⽅法抛出异常的时候

    @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影响原始⽅法的返回值

    原始⽅法的返回值,直接作为invoke⽅法的返回值返回,MethodInterceptor不会影响原始⽅法的返回值
    MethodInterceptor影响原始⽅法的返回值
    Invoke⽅法的返回值,不要直接返回原始⽅法的运⾏结果即可

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable{
        System.out.println("------log-----");
        Object ret = invocation.proceed();
        return false;//不要直接返回原始⽅法的运⾏结果即可
    }
    

2. 切⼊点详解

切⼊点决定额外功能加⼊位置(⽅法)

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

exection( (…)) —> 匹配了所有⽅法 a b c

  1. execution() 切⼊点函数
  2. * *(…) 切⼊点表达式

2.1 切⼊点表达式

  1. ⽅法切⼊点表达式

    * *(..) --> 所有⽅法
    
    * ---> 修饰符 返回值
    * ---> ⽅法名
    ()---> 参数表
    ..---> 对于参数没有要求 (参数有没有,参数有⼏个都⾏,参数是什么类型的都⾏)
    
    • 定义login⽅法作为切⼊点

      * login(..)
      
      # 定义 register 作为切⼊点
      * register(..)
      
    • 定义login⽅法且login⽅法有两个字符串类型的参数 作为切⼊点

      * login(String,String)
      
      # # 注意:⾮ java.lang 包中的类型,必须要写 全限定名
      * register(com.achang.proxy.User)
      
      # .. 可以和具体的参数类型连⽤
      * login(String,..) -->
      login(String),login(String,String),login(String,com.baizhiedu.proxy.User)
      
    • 精准⽅法切⼊点限定

      修饰符 返回值 包.类.⽅法(参数)
      
      * com.achang.proxy.UserServiceImpl.login(..)
      
      * com.achang.proxy.UserServiceImpl.login(String,String)
      
  2. 切⼊点·

    指定特定类作为切⼊点(额外功能加⼊的位置),⾃然这个类中的所有⽅法,都会加上对应的额外功能

    • 语法1

       # 类中的所有⽅法加⼊了额外功能
      * com.achang.proxy.UserServiceImpl.*(..)
      
    • 语法2

       # 忽略包
      1. 类只存在 ⼀级 包 com.UserServiceImpl
      * *.UserServiceImpl.*(..)
      2. 类存在 多级 包 com.achang.proxy.UserServiceImpl
      * *..UserServiceImpl.*(..)
      

  3. 切⼊点表达式 实战

    指定包作为额外功能加⼊的位置,⾃然包中的所有类及其⽅法都会加⼊额外的功能

    • 语法1

      # # 切⼊点包中的所有类,必须在 proxy 中,不能在 proxy 包的⼦包中
      * com.achang.proxy.*.* (..)
      
    • 语法2

       # 切⼊点当前包及其⼦包都⽣效
      * com.achang.proxy..*.* (..)
      

2.2 切⼊点函数

切⼊点函数⽤于执⾏切⼊点表达式

  1. execution

    最为重要的切⼊点函数,功能最全。

    • 执⾏⽅法切⼊点表达式、类切⼊点表达式、包切⼊点表达式
    弊端:execution执⾏切⼊点表达式 ,书写麻烦
    execution(* com.baizhiedu.proxy..*.*(..))
    注意:其他的切⼊点函数 简化是execution书写复杂度,功能上完全⼀致
    
  2. args

    作⽤:主要⽤于函数(⽅法) 参数的匹配
    切⼊点:⽅法参数必须得是2个字符串类型的参数
    execution( * * (String,String))
    args(String,String)
    
  3. within

    作⽤:主要⽤于进⾏类、包切⼊点表达式的匹配
    切⼊点:UserServiceImpl这个类
    
    execution( * * ..UserServiceImpl. *(..))
    within( *..UserServiceImpl)
    
    execution( * com.achang.proxy..* . *(..))
    within(com.achang.proxy..*)
    
  4. @annotation

    作⽤:为具有特殊注解的⽅法加⼊额外功能

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

    组装中的arround,自定义注解的实现,通过实现MethodInterceptor;pc是自定义注解

  5. 切⼊点函数的逻辑运算

    指的是 整合多个切⼊点函数⼀起配合⼯作,进⽽完成更为复杂的需求

    • and与操作

      案例:login 同时 参数 2个字符串
      1. execution( * login(String,String))
      2. execution( * login(..)) and args(String,String)
      
      注意:与操作不同⽤于同种类型的切⼊点函数
      
      案例:register⽅法 和 login⽅法作为切⼊点 xxxxxxxxxxxxxxxxxxxx
      execution( * login(..)) or execution(* register(..))
      
    • or或操作

      案例:register⽅法 和 login⽅法作为切⼊点
      execution( * login(..)) or execution(* register(..))
      

四、AOP编程

1. AOP概念

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

  • AOP (Aspect Oriented Programing)

    ⾯向切⾯编程 = Spring动态代理开发
    以切⾯为基本单位的程序开发,通过切⾯间的彼此协同,相互调⽤,完成程序的构建
    切⾯ = 切⼊点 + 额外功能

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

  • POP (Producer Oriented Programing) ⾯向过程(⽅法、函数)编程 C
    以过程为基本单位的程序开发,通过过程间的彼此协同,相互调⽤,完成程序的构建


2. AOP编程的开发步骤

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

3. 切⾯的名词解释

  • 切⾯ = 切⼊点 + 额外功能
    ⼏何学
  • = 点 + 相同的性质


五、AOP的底层实现原理

1. 核⼼问题

  1. AOP如何创建动态代理类(动态字节码技术)
  2. Spring⼯⼚如何加⼯创建代理对象
    通过原始对象的id值,获得的是代理对象

2. 动态代理类的创建

2.1 JDK的动态代理

  • Proxy.newProxyInstance⽅法参数详解

  • 编码
public class TestJDKProxy {
    /*
        1. 借⽤类加载器 TestJDKProxy、UserServiceImpl
        2. JDK8.x前
        final UserService userService = new UserServiceImpl();
        */
    public static void main(String[] args) {
        //1 创建原始对象
        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("------proxy log --------");
                //原始⽅法运⾏
                Object ret = method.invoke(userService, args);
                return ret;
            }
        };
        UserService userServiceProxy =(UserService)Proxy.newProxyInstance(UserServiceImpl.class.getClassLoader(),userService.getClass().getInterfaces(),handler);
        userServiceProxy.login("achang", "123456");
        userServiceProxy.register(new User());
    }
}

2.2 CGlib的动态代理

CGlib创建动态代理的原理:⽗⼦继承关系创建代理对象,原始类作为⽗类,代理类作为⼦类,这样既可以保证2者⽅法⼀致,同时在代理类中提供新的实现(额外功能+原始⽅法)

  • CGlib编码
package com.achang.cglib;

import com.achang.proxy.User;
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 TestCglib {
    public static void main(String[] args) {
        //1 创建原始对象
        UserService userService = new UserService();
        /*
    2 通过cglib⽅式创建动态代理对象
        Proxy.newProxyInstance(classloader,interface,invocationhandler)
        
        Enhancer.setClassLoader()
        Enhancer.setSuperClass()
        Enhancer.setCallback(); ---> MethodInterceptor(cglib)
        Enhancer.create() ---> 代理
    */
        Enhancer enhancer = new Enhancer();
        enhancer.setClassLoader(TestCglib.class.getClassLoader());
        enhancer.setSuperclass(userService.getClass());
        MethodInterceptor interceptor = new MethodInterceptor() {
            //等同于 InvocationHandler --- invoke
            @Override
            public Object intercept(Object o, Method method,Object[] args, MethodProxy methodProxy) throws Throwable {
                System.out.println("---cglib log----");
                Object ret = method.invoke(userService, args);
                return ret;
            }
        };
        enhancer.setCallback(interceptor);
        UserService userServiceProxy = (UserService)enhancer.create();
        
        userServiceProxy.login("achang", "1233456");
        userServiceProxy.register(new User());
    }
}
  • 总结
  1. JDK动态代理 Proxy.newProxyInstance() 通过接⼝创建代理的实现类
  2. Cglib动态代理 Enhancer 通过继承⽗类创建的代理类

3. Spring⼯⼚如何加⼯原始对象

  • 思路分析

  • 编码
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 handler = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method,Object[] args) throws Throwable 以上是关于Day381.AOP编程 -Spring5的主要内容,如果未能解决你的问题,请参考以下文章

##DAY13——可视化编程之XIB

day20-网络编程

day20-网络编程

day20-网络编程

泛化编程 day02

Day10 Python网络编程 Socket编程