SSM中的Spring框架AOP实操(第九课)

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: SSM中的Spring框架AOP实操(第九课)

第一部分:Spring配置文件applicationContext.xml文件写Aop的三种方式:

本博文是来对SpringAop中三种方式的回顾的

SpringAop的理论概念

1、什么是aop:

AOP(Aspect Oriented Programming)称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,是个比较经典的例子。


在不改变原有的逻辑的基础上,增加一些额外的功能。代理也是这个功能,读写分离也能用aop来做。


AOP可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善。OOP引入封装、继承、多态等概念来建立一种对象层次结构,用于模拟公共行为的一个集合。不过OOP允许开发者定义纵向的关系,但并不适合定义横向的关系,例如日志功能。日志代码往往横向地散布在所有对象层次中,而与它对应的对象的核心功能毫无关系对于其他类型的代码,如安全性、异常处理和透明的持续性也都是如此,这种散布在各处的无关的代码被称为横切(cross cutting),在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。


AOP技术恰恰相反,它利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。


使用"横切"技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事物。AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。

2、AOP的相关概念:

(1)横切关注点:对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点

(2)Aspect(切面):通常是一个类,里面可以定义切入点和通知

(3)JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用。被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器

(4)Advice(通知):AOP在特定的切入点上执行的增强处理,有before(前置),after(后置),afterReturning(最终),afterThrowing(异常),around(环绕)

(5)Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式

(6)weave(织入):将切面应用到目标对象并导致代理对象创建的过程

(7)introduction(引入):在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段

(8)AOP代理(AOP Proxy):AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类

(9)目标对象(Target Object): 包含连接点的对象。也被称作被通知或被代理对象。POJO

3、Advice通知类型介绍:

(1)Before:在目标方法被调用之前做增强处理,@Before只需要指定切入点表达式即可


(2)AfterReturning:在目标方法正常完成后做增强,@AfterReturning除了指定切入点表达式后,还可以指定一个返回值形参名returning,代表目标方法的返回值


(3)AfterThrowing:主要用来处理程序中未处理的异常,@AfterThrowing除了指定切入点表达式后,还可以指定一个throwing的返回值形参名,可以通过该形参名


来访问目标方法中所抛出的异常对象


(4)After:在目标方法完成之后做增强,无论目标方法时候成功完成。@After可以指定一个切入点表达式


(5)Around:环绕通知,在目标方法完成前后做增强处理,环绕通知是最重要的通知类型,像事务,日志等都是环绕通知,注意编程中核心是一个ProceedingJoinPoint

4、AOP使用场景:

Authentication 权限

Caching 缓存

Context passing 内容传递

Error handling 错误处理

Lazy loading 懒加载

Debugging  调试

logging, tracing, profiling and monitoring 记录跟踪 优化 校准

Performance optimization 性能优化

Persistence  持久化

Resource pooling 资源池

Synchronization 同步

Transactions 事务

5 注解的方式开发 Aop

在XML配置文件中对Bean的注册  

<bean id="userservice" class="com.service.UserServiceImp"></bean>
    <bean id="Car" class="com.service.CarImp"></bean>
    <bean id="Dog" class="com.service.DogImp"></bean>
    <bean id="Firsh" class="com.service.FirshImp"></bean>
    <bean id="Pig" class="com.service.PigImp"></bean>
    <bean id="log" class="log.Log"></bean>
    <bean id="afterlog" class="log.Afterlog"></bean>

在配置文件中写上注解的方式

方案一 注解的方式

    <!--&lt;!&ndash;方式三 SpringAop的注解的方式&ndash;&gt;-->
        <bean id="annotationPointCut" class="diy.AnnotationPointCut"></bean>
        <!--代理Aop注解支持 JDK  proxy-target-class="false" -->
        <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
    <bean id="annotationPointCut" class="diy.DogAnnotationPointCut"></bean>
    <!--代理Aop注解支持 JDK  proxy-target-class="false" -->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

自己创建一个类在类中写关于SpringAop注解

package diy;
import com.sun.istack.internal.NotNull;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class AnnotationPointCut {
    @Before("execution(* com.service.UserServiceImp.*(..))")
    public void before() {
        System.out.println("==============方法执行前=================");
    }
    @After("execution(* com.service.UserServiceImp.*(..))")
    public  void  seeyou(){
        System.out.println("游湖调用的seeyou方法");
    }
    @After("execution(* com.service.UserServiceImp.*(..))")
    public void after() {
        System.out.println("==============方法执行后==================");
    }
    @Around("execution(* com.service.UserServiceImp.*(..))")
    public  void  around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("在环绕增强中,我们可以定义一个参数,代表我们要获取处理的切入点");
        Object o = jp.proceed();
        System.out.println("环绕前");
    }
}
package diy;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class DogAnnotationPointCut {
    @Before("execution(* com.service.DogImp.*(..))")
    public void before() {
        System.out.println("==============方法执行前=================");
    }
    @After("execution(* com.service.DogImp.*(..))")
    public  void  seeyou(){
        System.out.println("游湖调用的seeyou方法");
    }
    @After("execution(* com.service.DogImp.*(..))")
    public void after() {
        System.out.println("==============方法执行后==================");
    }
    @Around("execution(* com.service.DogImp.*(..))")
    public  void  around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("在环绕增强中,我们可以定义一个参数,代表我们要获取处理的切入点");
        Object o = jp.proceed();
        System.out.println("环绕前");
    }
}

6 自定义实体类的方式开发SpringAop在北至文件中书写的内容


方案二 自定义实体类

<!--    SpringAop的方案二 Class:要在那个类中去定义实体呢-->
        <bean id="diy" class="diy.DiyPointcut"></bean>
        <aop:config>
            <!--自定义切面 ref 要自定义实体类的包名-->
            <aop:aspect ref="diy">
                <!--切入点 实现类的方法在哪里 -->
               <aop:pointcut id="point" expression="execution(* com.service.UserServiceImp.*(..)))"/>
                <!--通知将自己定义的方法定义到本类中来-->
                <aop:before method="before" pointcut-ref="point"></aop:before>
                <aop:after method="after" pointcut-ref="point"></aop:after>
                <aop:after method="see" pointcut-ref="point"></aop:after>
            </aop:aspect>
        </aop:config>

自己创建一个类在类中写关于SpringAop 方法

package diy;
/**
 * 自定义类
 */
public class DiyPointcut {
    public  void  before(){
        System.out.println("==============方法执行前=================");
    }
    public  void  after(){
        System.out.println("==============方法执行后==================");
    }
    /**
     * 在配置文件中注册方法
     */
    public  void  see(){
        System.out.println("===========SpringAOP第二种方式自定义类的方式=============");
    }
}

7 原始方案 SpringAop

package log;
import org.springframework.aop.AfterReturningAdvice;
import java.lang.reflect.Method;
public class Afterlog  implements AfterReturningAdvice {
    @Override
    public void afterReturning(Object o, Method method, Object[] objects, Object o1) throws Throwable {
        System.out.println("执行了"+method.getName()+"返回了"+o1);
    }
}
package log;
import org.springframework.aop.MethodBeforeAdvice;
import java.lang.reflect.Method;
public class Log  implements MethodBeforeAdvice {
    //Method 要执行的目标对象方法
    //orgs  参数
    //target 目标对象
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println(target.getClass().getName()+"的"+method.getName()+"被执行了");
    }
}

方案三:原始方案

<!--Car的注册-->
    <aop:config>
        <!--切入点  execution表达式( 要执行的位置! * * * *)-->
        <aop:pointcut id="pointcut" expression="execution(* com.service.CarImp.*(..))"/>
        <!--执行环绕增加-->
        <aop:advisor advice-ref="log" pointcut-ref="pointcut"></aop:advisor>
        <aop:advisor advice-ref="afterlog" pointcut-ref="pointcut"></aop:advisor>
    </aop:config>

第二部分代码测试:

import com.service.*;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class MaTest {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
    //动态代理的是接口
        UserService userservice = (UserService) applicationContext.getBean("userservice");
        userservice.update();
        System.out.println("==============================================================");
        //Car测试
        Car car =(Car) applicationContext.getBean("Car");
        car.run();
    //   Dog测试
        System.out.println("==============================================================");
        Dog dog=(Dog) applicationContext.getBean("Dog") ;
        dog.eat();
        dog.update();
        System.out.println("==============FirshImp=====================");
    //Firsh测试
        Firsh firsh=(Firsh) applicationContext.getBean("Firsh");
        firsh.update();
    }
}

8 代码模块

Car类:

package com.service;
public class CarImp implements Car {
    @Override
    public void add() {
        System.out.println("增加一条记录在Car在Car表中");
    }
    @Override
    public void delete() {
    }
    @Override
    public void update() {
    }
    @Override
    public void query() {
    }
    @Override
    public void X() {
    }
    @Override
    public void run() {
        System.out.println("Car在run中奔跑在Car的类中");
    }
}

Dog类:

package com.service;
public class DogImp  implements Dog{
    @Override
    public void eat() {
        System.out.println("DogImp在吃食物");
    }
    @Override
    public void add() {
        System.out.println("DogImp在吃食物在增加记录");
    }
    @Override
    public void delete() {
        System.out.println("DogImp在删除信息");
    }
    @Override
    public void update() {
        System.out.println("DogImp在修改信息");
    }
    @Override
    public void query() {
    }
    @Override
    public void X() {
    }
}

Firsh类:

标题

package com.service;
public class FirshImp  implements Firsh{
    @Override
    public void swwing() {
        System.out.println("FirshIm在Swwing游泳");
    }
    @Override
    public void add() {
        System.out.println("FirshImp增加的操作");
    }
    @Override
    public void delete() {
        System.out.println("FirshImp删除的操作");
    }
    @Override
    public void update() {
        System.out.println("FirshImp修改的操作");
    }
    @Override
    public void query() {
    }
    @Override
    public void X() {
    }
}

Pig类

package com.service;
public class PigImp implements pig {
    @Override
    public void sleep() {
        System.out.println("PigImp自己最喜欢睡觉");
    }
    @Override
    public void add() {
        System.out.println("用户开始从某某数据库中的表中增加了一个用户记录");
    }
    @Override
    public void delete() {
        System.out.println("用户开始从某某数据库中的表中删除了一个用户");
    }
    @Override
    public void update() {
        System.out.println("用户开始从某某数据库中的表中修改了一条记录");
    }
    @Override
    public void query() {
        System.out.println("用户开始从某某数据库中的表中查询用户多条信息和单条记录");
    }
    @Override
    public void X() {
        System.out.println("本人自定义的方法为X()方法");
    }
    @Override
    public void pigpig() {
        System.out.println("PigImp自己独有的方法构造");
    }
}

UserService类:

package com.service;
public interface UserService {
    //在 UserService定义 增删改查的方法 来构造
    public  void  add();
    public  void  delete();
    public  void  update();
    public  void  query();
    public  void X();
}
package com.service;
public class UserServiceImp  implements UserService{
    /**
     *  UserServiceImp  implements UserService
     *  在这个类中实现UserService中没有实现的方法
     */
    @Override
    public void add() {
        System.out.println("用户开始从某某数据库中的表中增加了一个用户记录");
    }
    @Override
    public void delete() {
        System.out.println("用户开始从某某数据库中的表中删除了一个用户");
    }
    @Override
    public void update() {
        System.out.println("用户开始从某某数据库中的表中修改了一条记录");
    }
    @Override
    public void query() {
        System.out.println("用户开始从某某数据库中的表中查询用户多条信息和单条记录");
    }
    @Override
    public void X() {
        System.out.println("本人自定义的方法为X()方法");
    }
}
相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
4天前
|
XML Java 测试技术
Spring5入门到实战------17、Spring5新功能 --Nullable注解和函数式注册对象。整合JUnit5单元测试框架
这篇文章介绍了Spring5框架的三个新特性:支持@Nullable注解以明确方法返回、参数和属性值可以为空;引入函数式风格的GenericApplicationContext进行对象注册和管理;以及如何整合JUnit5进行单元测试,同时讨论了JUnit4与JUnit5的整合方法,并提出了关于配置文件加载的疑问。
Spring5入门到实战------17、Spring5新功能 --Nullable注解和函数式注册对象。整合JUnit5单元测试框架
|
4天前
|
Java
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
这篇文章是Spring5框架的实战教程,深入讲解了AOP的基本概念、如何利用动态代理实现AOP,特别是通过JDK动态代理机制在不修改源代码的情况下为业务逻辑添加新功能,降低代码耦合度,并通过具体代码示例演示了JDK动态代理的实现过程。
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
|
4天前
|
XML Java 数据格式
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
这篇文章是Spring5框架的AOP切面编程教程,通过XML配置方式,详细讲解了如何创建被增强类和增强类,如何在Spring配置文件中定义切入点和切面,以及如何将增强逻辑应用到具体方法上。文章通过具体的代码示例和测试结果,展示了使用XML配置实现AOP的过程,并强调了虽然注解开发更为便捷,但掌握XML配置也是非常重要的。
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
|
1天前
|
安全 前端开发 Java
随着企业应用复杂度提升,Java Spring框架以其强大与灵活特性简化开发流程,成为构建高效、可维护应用的理想选择
随着企业应用复杂度提升,Java Spring框架以其强大与灵活特性简化开发流程,成为构建高效、可维护应用的理想选择。依赖注入使对象管理交由Spring容器处理,实现低耦合高内聚;AOP则分离横切关注点如事务管理,增强代码模块化。Spring还提供MVC、Data、Security等模块满足多样需求,并通过Spring Boot简化配置与部署,加速微服务架构构建。掌握这些核心概念与工具,开发者能更从容应对挑战,打造卓越应用。
6 1
|
3天前
|
运维 Java Nacos
Spring Cloud应用框架:Nacos作为服务注册中心和配置中心
Spring Cloud应用框架:Nacos作为服务注册中心和配置中心
|
4天前
|
XML Java Maven
Spring5入门到实战------16、Spring5新功能 --整合日志框架(Log4j2)
这篇文章是Spring5框架的入门到实战教程,介绍了Spring5的新功能——整合日志框架Log4j2,包括Spring5对日志框架的通用封装、如何在项目中引入Log4j2、编写Log4j2的XML配置文件,并通过测试类展示了如何使用Log4j2进行日志记录。
Spring5入门到实战------16、Spring5新功能 --整合日志框架(Log4j2)
|
4天前
|
Java API Spring
Spring5入门到实战------1、Spring5框架概述、入门案例
这篇文章是Spring5框架的入门教程,概述了Spring框架的核心概念和特点,并通过一个创建普通Java类的案例,详细演示了从下载Spring核心Jar包、创建配置文件、编写测试代码到运行测试结果的完整流程,涵盖了Spring IOC容器的使用和依赖注入的基本用法。
Spring5入门到实战------1、Spring5框架概述、入门案例
|
6天前
|
Java Spring
Spring的AOP组件详解
该文章主要介绍了Spring AOP(面向切面编程)组件的实现原理,包括Spring AOP的基础概念、动态代理模式、AOP组件的实现以及Spring选择JDK动态代理或CGLIB动态代理的依据。
Spring的AOP组件详解
|
4天前
|
XML Java 数据库
Spring5入门到实战------10、操作术语解释--Aspectj注解开发实例。AOP切面编程的实际应用
这篇文章是Spring5框架的实战教程,详细解释了AOP的关键术语,包括连接点、切入点、通知、切面,并展示了如何使用AspectJ注解来开发AOP实例,包括切入点表达式的编写、增强方法的配置、代理对象的创建和优先级设置,以及如何通过注解方式实现完全的AOP配置。
|
4天前
|
Java API Spring
Spring5入门到实战------1、Spring5框架概述、入门案例
这篇文章是Spring5框架的入门教程,概述了Spring框架的核心概念和特点,并通过一个创建普通Java类的案例,详细演示了从下载Spring核心Jar包、创建配置文件、编写测试代码到运行测试结果的完整流程,涵盖了Spring IOC容器的使用和依赖注入的基本用法。