《spring上课笔记》---class3---AOP(面向切面编程)

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 《spring上课笔记》---class3---AOP(面向切面编程)

一:代理模式


代理模式简介:


代理模式是GoF23种常用设计模式之一,可以使用代理模式创建代理对象,让代理对象控制目标对象的访问,并且可以在不改变目标对象原有的逻辑功能之下添加一些额外的功能。


案例分析:


实现一个用户注册功能,现在要在原有注册功能之上添加一个打印日志的功能。为了不破坏原有的业务逻辑,我们可以使用代理模式。


20210317133544727.png

1.静态代理实现:


代理对象与被代理对象必须实现同一接口,在代理对象中实现日志打印服务。

  • UserSevice接口
public interface UserService {
  public boolean login(String userName,String password);
}


  • UserService实现类:
public class UserServiceImpl implements UserService{
  @Override
  public boolean login(String userName, String password) {
    // TODO Auto-generated method stub
    boolean result="zs".equals(userName)&&"123".equals(password);
    System.out.println("原有业务执行结果是:"+result);
    return result;
  }
}
  • 打印日志工具类
public class LogUtil {
  public static void log() {
    System.out.println("日志:"+new Date().toLocaleString());
  }
}
  • UserService代理类
public class UserProxy implements UserService{
  private UserService userService;
  @Override
  public boolean login(String userName, String password) {
    userService=new UserServiceImpl();
    boolean result=userService.login(userName, password);
    if(result) {
      LogUtil.log();
    }
    return result;
  }
}
  • 测试类
public class Test {
   public static void main(String[] args) {
  UserProxy userProxy=new UserProxy();
  System.out.println(userProxy.login("zs", "123"));
}
}


  • 缺点:代理对象的接口只服务于一种类型的对象,有局限性

2.动态代理实现:

JDK1.3之后加入了实现动态代理的API,代理类实现InvocationHandler的接口就行。

  • 代理类:
public class LoggerHandler implements InvocationHandler{
    private Object delegate;
    public   Object bind(Object delegate) {
      this.delegate=delegate;
      return Proxy.newProxyInstance(delegate.getClass().getClassLoader(), delegate.getClass().getInterfaces(), this);
    }
  @Override
  public Object invoke(Object arg0, Method arg1, Object[] arg2) throws Throwable {
    System.out.println(arg0.getClass().getName());//代理对象
    System.out.println(arg1.getName());//代理对象的方法
    System.out.println(arg2.toString());//方法中的参数
    Object result=null;
    result=arg1.invoke(delegate, arg2);
    LogUtil.log();
    return result;
  }
}


  • 测试类:
public class Test {
   public static void main(String[] args) {
  UserService us=new UserServiceImpl();
  UserService proxy=(UserService) new LoggerHandler().bind(us);
  proxy.login("zs", "123");
}
}

二:AOP概述


AOP(Aspect Oriented Programming)意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。

AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。

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


1.SpringAPI传统方式实现AOP


切面通过实现一些接口,来达到通知的作用。分为前置通知,后置通知,环绕通知,异常通知,最终通知。


业务类接口:

package com.user.service;
public interface UserService {
  public boolean login(String userName, String pwd);
  public boolean regist(String userName, String pwd);
}

业务实现类:

public class UserServiceImpl implements UserService {
  @Override
  public boolean login(String userName, String pwd) {
    System.out.println("目标对象中,账号密码是:" + userName + ":" + pwd);
    boolean result = "zs".equals(userName) && "202cb962ac59075b964b07152d234b70".equals(pwd);
    System.out.println("用户登录");
    try {
      Thread.sleep(2000);
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
    int n = 10 / 0;
    return result;
  }
  @Override
  public boolean regist(String userName, String pwd) {
    System.out.println("目标对象中,账号密码是:" + userName + ":" + pwd);
    System.out.println("用户注册成功");
    return true;
  }
}

前置通知:

public class Md5Advice implements MethodBeforeAdvice {
  @Override
  public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
    // TODO Auto-generated method stub
    System.out.println(arg0.getName());// 方法名称
    System.out.println(arg1.toString());// 方法参数
    System.out.println(arg2);// 被代理对象
    String newPwd = Md5Encode.getMD5(arg1[1].toString().getBytes());
    arg1[1] = newPwd;
  }
}

后置通知:

public class ScoreAdvice implements AfterReturningAdvice {
  @Override
  public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {
    System.out.println(arg0);// 方法返回的结果
    System.out.println(arg1.getName());// 方法名
    System.out.println(arg3);// 被代理对象
    System.out.println("后置通知:用户登录成功,增加10个积分");
  }
}

环绕通知:

public class TimeAdvice implements MethodInterceptor {
  /**
   * 环绕通知
   */
  @Override
  public Object invoke(MethodInvocation arg0) throws Throwable {
    System.out.println("methodbefore:" + new Date().toLocaleString());
    Object result = arg0.proceed();// 目标方法是否执行
    System.out.println("methodafter:" + new Date().toLocaleString());
    return result;
  }
}

异常通知:

public class LoginExceptionAdvice implements ThrowsAdvice {
  public void afterThrowing(Exception e) {
    e.printStackTrace();
    System.out.println("异常通知");
  }
}

配置文件:

<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:context="http://www.springframework.org/schema/context"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
  <bean id="UserServiceImpl"
    class="com.user.service.UserServiceImpl"></bean>
  <bean id="Md5Advice" class="com.advice.Md5Advice"></bean>
  <bean id="ScoreAdvice" class="com.advice.ScoreAdvice"></bean>
  <bean id="TimeAdvice" class="com.advice.TimeAdvice"></bean>
  <bean id="LoginExceptionAdvice"
    class="com.advice.LoginExceptionAdvice"></bean>
  <bean id="UserServiceProxy"
    class="org.springframework.aop.framework.ProxyFactoryBean">
    <property name="proxyInterfaces"
      value="com.user.service.UserService"></property>
    <property name="target" ref="UserServiceImpl"></property>
    <property name="interceptorNames">
      <list>
        <!-- <value>Md5Advice</value> <value>ScoreAdvice</value> -->
        <value>TimeAdvice</value>
        <value>LoginExceptionAdvice</value>
      </list>
    </property>
  </bean>
</beans>

测试类:

public class Test {
  public static void main(String[] args) {
    ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
    UserService us = (UserService) ctx.getBean("UserServiceProxy");
    us.login("zs", "123");
    // us.regist("zs", "123");
  }
}

2.Spring基于xml方式实现AOP

业务类:

public class Test {
  public static void main(String[] args) {
    ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
    UserService us = (UserService) ctx.getBean("UserServiceProxy");
    us.login("zs", "123");
    // us.regist("zs", "123");
  }
}

通知类:

public class BeforeAdvice {
  /**
   * 切点 前置通知
   * 
   * @param joinPoint
   */
  public void methodBefore(JoinPoint joinPoint) {
    System.out.println("前置通知");
    System.out.println(joinPoint.getSignature().getName());// 代理方法名
    System.out.println(joinPoint.getArgs());// 参数数组
    System.out.println(joinPoint.getTarget());// 被代理对象
    System.out.println("================");
  }
  /**
   * 后置通知
   * 
   * @param joinPoint
   * @param result
   */
  public void afterReturning(JoinPoint joinPoint, Object result) {
    System.out.println("后置通知");
    System.out.println("================");
  }
  /**
   * 环绕通知
   * 
   * @param joinPoint
   * @return
   */
  public Object aroundMethod(ProceedingJoinPoint joinPoint) {
    System.out.println("环绕通知开始:");
    System.out.println("================");
    Object[] args = joinPoint.getArgs();// 方法参数
    args[0] = args[0] + "456";
    Object result = null;
    try {
      result = joinPoint.proceed(args);// 原方法是否执行
    } catch (Throwable e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    System.out.println("环绕通知结束");
    System.out.println("================");
    return result;
  }
  /**
   * 异常通知
   * 
   * @param e
   */
  public void throwExMethod(Exception e) {
    System.out.println("异常通知开始");
    System.out.println("异常信息:" + e.getMessage());
    System.out.println("================");
  }
  /**
   * 最终通知
   */
  public void afterMethod(JoinPoint jointPoint) {
    System.out.println("最终通知");
    System.out.println("================");
  }
}

配置文件:

<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:context="http://www.springframework.org/schema/context"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
      http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">
  <bean id="UserService" class="com.UserService"></bean>
  <bean id="MyAdvice" class="Advice.BeforeAdvice"></bean>
  <aop:config>
    <aop:aspect id="MyAspect" ref="MyAdvice">
      <aop:before method="methodBefore"
        pointcut="execution(* com.*.*(..))" /> <!-- com下的所有类型,所有参数 -->
      <aop:after-returning method="afterReturning"
        pointcut="execution(* com.*.*(..))" returning="result" />
      <aop:around method="aroundMethod"
        pointcut="execution(* com.*.*(..))" />
      <aop:after-throwing method="throwExMethod"
        pointcut="execution(* com.*.*(..))" throwing="e" />
      <aop:after method="afterMethod"
        pointcut="execution(* com.*.*(..))" />
    </aop:aspect>
  </aop:config>
</beans>

测试类:

public class Test {
  public static void main(String[] args) {
    ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
    UserService us = (UserService) ctx.getBean("UserService");
    us.login("zs", "123");
  }
}

3.Spring基于注解方式实现AOP

业务类:

@Component("UserService")
public class UserService {
  public boolean login(String userName, String passWord) {
    System.out.println("登录名是:" + userName);
    System.out.println("密码是:" + passWord);
    System.out.println("==============");
    int n = 10 / 0;
    return true;
  }
}

通知类:

@Component
@Aspect
public class BeforeAdvice {
  /**
   * 切点 前置通知
   * 
   * @param joinPoint
   */
  @Before("execution(* com.*.*(..))")
  public void methodBefore(JoinPoint joinPoint) {
    System.out.println("前置通知");
    System.out.println(joinPoint.getSignature().getName());// 代理方法名
    System.out.println(joinPoint.getArgs());// 参数数组
    System.out.println(joinPoint.getTarget());// 被代理对象
    System.out.println("================");
  }
  /**
   * 后置通知
   * 
   * @param joinPoint
   * @param result
   */
  @AfterReturning(pointcut = "execution(* com.*.*(..))", returning = "result")
  public void afterReturning(JoinPoint joinPoint, Object result) {
    System.out.println("后置通知");
    System.out.println("================");
  }
  /**
   * 环绕通知
   * 
   * @param joinPoint
   * @return
   */
  @Around("execution(* com.*.*(..))")
  public Object aroundMethod(ProceedingJoinPoint joinPoint) {
    System.out.println("环绕通知开始:");
    System.out.println("================");
    Object[] args = joinPoint.getArgs();// 方法参数
    args[0] = args[0] + "456";
    Object result = null;
    try {
      result = joinPoint.proceed(args);// 原方法是否执行
    } catch (Throwable e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    System.out.println("环绕通知结束");
    System.out.println("================");
    return result;
  }
  /**
   * 异常通知
   * 
   * @param e
   */
  @AfterThrowing(pointcut = "execution(* com.*.*(..))", throwing = "e")
  public void throwExMethod(Exception e) {
    System.out.println("异常通知开始");
    System.out.println("异常信息:" + e.getMessage());
    System.out.println("================");
  }
  /**
   * 最终通知
   */
  @After("execution(* com.*.*(..))")
  public void afterMethod(JoinPoint jointPoint) {
    System.out.println("最终通知");
    System.out.println("================");
  }
}

配置文件:

<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:context="http://www.springframework.org/schema/context"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
      http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">
  <bean id="UserService" class="com.UserService"></bean>
  <bean id="MyAdvice" class="Advice.BeforeAdvice"></bean>
  <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
  <context:component-scan base-package="com"></context:component-scan>
</beans>

测试类:

public class Test {
  public static void main(String[] args) {
    ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
    UserService us = (UserService) ctx.getBean("UserService");
    us.login("zs", "123");
  }
}
相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
20天前
Micronaut AOP与代理机制:实现应用功能增强,无需侵入式编程的秘诀
AOP(面向切面编程)能够帮助我们在不修改现有代码的前提下,为应用程序添加新的功能或行为。Micronaut框架中的AOP模块通过动态代理机制实现了这一目标。AOP将横切关注点(如日志记录、事务管理等)从业务逻辑中分离出来,提高模块化程度。在Micronaut中,带有特定注解的类会在启动时生成代理对象,在运行时拦截方法调用并执行额外逻辑。例如,可以通过创建切面类并在目标类上添加注解来记录方法调用信息,从而在不侵入原有代码的情况下增强应用功能,提高代码的可维护性和可扩展性。
43 1
|
12天前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
2天前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
26 9
|
1月前
Micronaut AOP与代理机制:实现应用功能增强,无需侵入式编程的秘诀
【9月更文挑战第9天】AOP(面向切面编程)通过分离横切关注点提高模块化程度,如日志记录、事务管理等。Micronaut AOP基于动态代理机制,在应用启动时为带有特定注解的类生成代理对象,实现在运行时拦截方法调用并执行额外逻辑。通过简单示例展示了如何在不修改 `CalculatorService` 类的情况下记录 `add` 方法的参数和结果,仅需添加 `@Loggable` 注解即可。这不仅提高了代码的可维护性和可扩展性,还降低了引入新错误的风险。
38 13
|
25天前
|
Java 数据库连接 API
【Java笔记+踩坑】Spring Data JPA
从常用注解、实体类和各层编写方法入手,详细介绍JPA框架在增删改查等方面的基本用法,以及填充用户名日期、分页查询等高级用法。
【Java笔记+踩坑】Spring Data JPA
|
25天前
|
Java 数据库连接 数据库
Spring基础3——AOP,事务管理
AOP简介、入门案例、工作流程、切入点表达式、环绕通知、通知获取参数或返回值或异常、事务管理
Spring基础3——AOP,事务管理
|
25天前
|
Java 数据库连接 数据格式
【Java笔记+踩坑】Spring基础2——IOC,DI注解开发、整合Mybatis,Junit
IOC/DI配置管理DruidDataSource和properties、核心容器的创建、获取bean的方式、spring注解开发、注解开发管理第三方bean、Spring整合Mybatis和Junit
【Java笔记+踩坑】Spring基础2——IOC,DI注解开发、整合Mybatis,Junit
|
2月前
|
缓存 Java 开发者
Spring高手之路22——AOP切面类的封装与解析
本篇文章深入解析了Spring AOP的工作机制,包括Advisor和TargetSource的构建与作用。通过详尽的源码分析和实际案例,帮助开发者全面理解AOP的核心技术,提升在实际项目中的应用能力。
25 0
Spring高手之路22——AOP切面类的封装与解析
|
2月前
|
Java Spring XML
掌握面向切面编程的秘密武器:Spring AOP 让你的代码优雅转身,横切关注点再也不是难题!
【8月更文挑战第31天】面向切面编程(AOP)通过切面封装横切关注点,如日志记录、事务管理等,使业务逻辑更清晰。Spring AOP提供强大工具,无需在业务代码中硬编码这些功能。本文将深入探讨Spring AOP的概念、工作原理及实际应用,展示如何通过基于注解的配置创建切面,优化代码结构并提高可维护性。通过示例说明如何定义切面类、通知方法及其应用时机,实现方法调用前后的日志记录,展示AOP在分离关注点和添加新功能方面的优势。
40 0
|
2月前
|
Java Spring 容器
彻底改变你的编程人生!揭秘 Spring 框架依赖注入的神奇魔力,让你的代码瞬间焕然一新!
【8月更文挑战第31天】本文介绍 Spring 框架中的依赖注入(DI),一种降低代码耦合度的设计模式。通过 Spring 的 DI 容器,开发者可专注业务逻辑而非依赖管理。文中详细解释了 DI 的基本概念及其实现方式,如构造器注入、字段注入与 setter 方法注入,并提供示例说明如何在实际项目中应用这些技术。通过 Spring 的 @Configuration 和 @Bean 注解,可轻松定义与管理应用中的组件及其依赖关系,实现更简洁、易维护的代码结构。
37 0