Spring AOP详解

AOP:Aspect Oriented Programing :面向切面编程
当需要为分散的对象引入公共的行为时,我们就会考虑以前所接触的封装、继承和多态性等来建立一种对象层次结构,用以模拟公共行为的一个集合还可行不,答案是显而易见的,不行。所以我们就需要用到AOP
AOP将其分为两个部分:核心关注点和横切关注点
核心关注点: 业务处理的主要流程
横切关注点:与业务处理的主要流程关系不大的部分。横切关注点的一个特点是,它们经常发生在核心关注点的多处,而各处都基本相似。比如权限认
证、日志、事务处理。
AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。
那除了核心业务想增加一些功能怎么办?
首先我们来看一下静态代理和动态代理
静态代理:
在这里插入图片描述
如上图所示:代理对象除了完成公共的业务操作以外,还可以进行一些日志的打印,安全性检查等业务。
接下来我们看一下动态代理:
在这里插入图片描述
每次JVM根据目标对象的不同需求给它生成不同的虚拟的代理对象去完成业务。
代码实现:

//用于生成动态代理对象(反射)
public class DynamicProxy implements InvocationHandler {
    //引入真实的对象
    public Object targetObject;
    //创建一个方法,通过反射生成动态代理对象
    public Object createDynamicProxyObjectInstance(Object paramObj) {
        this.targetObject = paramObj;
        return Proxy.newProxyInstance(
                this.targetObject.getClass().getClassLoader(),
                this.targetObject.getClass().getInterfaces(),
                this);
    }
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
      Object resultObj=null;
      try{
          checkSecurity();
          resultObj=method.invoke(this.targetObject,args);
          return resultObj;
      }catch(Exception e){
          e.printStackTrace();
      }
      return resultObj;
    }
    public void checkSecurity(){
        System.out.println("checkSecurity");
    }
}

测试类:

 public static void main(String[] args) {
        DynamicProxy dynamicProxy=new DynamicProxy();
        IUserManager iUserManager= (IUserManager) dynamicProxy.createDynamicProxyObjectInstance(new UserManagerImpl());
        iUserManager.addUser("Sally","888");
        iUserManager.modifyUser(1,"Sally","666");
        iUserManager.delUser(1);
    }

说明: IUserManager iUserManager= (IUserManager) dynamicProxy.createDynamicProxyObjectInstance(new UserManagerImpl());
每次根据需求创建动态代理对象。
比以上更好的解决方案就是由AOP提供的
使用AOP的优点:
降低模块耦合度
使系统容易扩展
延迟设计决定:使用AOP,设计师可以推迟为将来的需求作决定,因为需求作为独立的方面很容易实现
有更好的代码复用性
AOP术语:
通知(advice),切点(pointcut)和连接点(joinpoint)。
通知:描述了切面要完成的工作(有5种类型)
前置通知(Before): 在目标方法被调用之前调用通知功能;
后置通知(After): 在目标方法完成之后调用通知,此时不会关心方法的输出是什么;
返回通知(After-returning):在目标方法成功执行之后调用通知;
异常通知(After-throwing): 在目标方法抛出异常后通知调用;
环绕通知(Around):通知包裹了被通知的方法,在被通知的方法通知之前和调用之后执行自定义的行为
在这里插入图片描述
如上图所示:
Advice就是通知,增强处理的功能,比如说ChecckSecurity等,定义好,什么时候想用了再使用。允许Advice的地方,基本每个方法的before,after,throws都可以是JoinPoint(连接点),即和方法有关的前后都是连接点。若一个类中,有很多个方法,但是并不想都使用Advice,只是想让其中的几个,在调用方法before,after,throws时干点什么,就可以用PointCut来定义方法,筛选连接点,选中我们想要的 ,将其织入(weaving)到目标对象中.
通知和切入点的结合就是Aspect(切面)。
使用注解创建切面
(1)添加依赖
将以下的maven jar包添加到项目中去。

 <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.1.5.RELEASE</version>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.springframework/spring-core -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>5.1.5.RELEASE</version>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.springframework/spring-beans -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-beans</artifactId>
            <version>5.1.5.RELEASE</version>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
            <version>5.1.5.RELEASE</version>
        </dependency>


        <!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>5.1.5.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>

        <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.4</version>
        </dependency>

(2)定义切面:

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

@Aspect
public class SecurityHandler {
    //PointCut标识一个范围
    @Pointcut("execution(* com.zh.*.*(..))")
private void allMethod(){}
//Advice[Before]
    //横切关注点
@Before("allMethod()")
public void checkSecurity(){
    System.out.println("=============checkSecurity()============");
}

}

(3)XML配置使用切面

<?xml version="1.0" encoding="ISO-8859-1"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:util="http://www.springframework.org/schema/util"
       xmlns:cache="http://www.springframework.org/schema/cache"
       xsi:schemaLocation="
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/cache
    http://www.springframework.org/schema/cache/spring-cache-3.1.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop.xsd
    http://www.springframework.org/schema/util
    http://www.springframework.org/schema/util/spring-util.xsd">

    <!--?spring??????-->
    <aop:aspectj-autoproxy/>

    <bean id="securityHandler" class="com.zh.SecurityHandler"/>

    <bean id="userManagerImpl" class="com.zh.UserManagerImpl"/>
</beans>

切点表达式定义:
arg() : 限制连接点匹配参数为指定类型的执行方法
@args(): 限制连接点匹配参数由指定注解标注的执行方法
execution() : 用于匹配时连接点的执行方法
this() : 限制连接点匹配AOP代理的bean引用为指定类型的类
target : 限制连接点匹配目标对象为指定类型的类
@target : 限制连接点匹配特定的执行对象,这些对象对应的类要具有指定类型的注解
within() : 限制连接点匹配指定的类型
@within() : 限制连接点匹配指定注解所标注的类型(当使用Spring AOP时,方法定义在由指定的注解所标注
的类里)
@annotation() : 限定匹配带有指定注解的连接点

动态和静态代理的完整源代码请看github链接:
https://github.com/zhdfuture/Spring2/commit/1ca3145e2b48b27e858ff8831810c41c0c16bb27
AOP实现的完整源代码请看github链接:
https://github.com/zhdfuture/Spring3/commit/d5c81114a14465248cdc5aa595fd39e4daf15d89

基于数据挖掘的音乐推荐系统设计与实现 需要一个代码说明,不需要论文 采用python语言,django框架,mysql数据库开发 编程环境:pycharm,mysql8.0 系统分为前台+后台模式开发 网站前台: 用户注册, 登录 搜索音乐,音乐欣赏(可以在线进行播放) 用户登陆时选择相关感兴趣的音乐风格 音乐收藏 音乐推荐算法:(重点) 本课题需要大量用户行为(如播放记录、收藏列表)、音乐特征(如音频特征、歌曲元数据)等数据 (1)根据用户之间相似性或关联性,给一个用户推荐与其相似或有关联的其他用户所感兴趣的音乐; (2)根据音乐之间的相似性或关联性,给一个用户推荐与其感兴趣的音乐相似或有关联的其他音乐。 基于用户的推荐和基于物品的推荐 其中基于用户的推荐是基于用户的相似度找出相似相似用户,然后向目标用户推荐其相似用户喜欢的东西(和你类似的人也喜欢**东西); 而基于物品的推荐是基于物品的相似度找出相似的物品做推荐(喜欢该音乐的人还喜欢了**音乐); 管理员 管理员信息管理 注册用户管理,审核 音乐爬虫(爬虫方式爬取网站音乐数据) 音乐信息管理(上传歌曲MP3,以便前台播放) 音乐收藏管理 用户 用户资料修改 我的音乐收藏 完整前后端源码,部署后可正常运行! 环境说明 开发语言:python后端 python版本:3.7 数据库:mysql 5.7+ 数据库工具:Navicat11+ 开发软件:pycharm
MPU6050是一款广泛应用在无人机、机器人和运动设备中的六轴姿态传感器,它集成了三轴陀螺仪和三轴加速度计。这款传感器能够实时监测并提供设备的角速度和线性加速度数据,对于理解物体的动态运动状态至关重要。在Arduino平台上,通过特定的库文件可以方便地与MPU6050进行通信,获取并解析传感器数据。 `MPU6050.cpp`和`MPU6050.h`是Arduino库的关键组成部分。`MPU6050.h`是头文件,包含了定义传感器接口和函数声明。它定义了类`MPU6050`,该类包含了初始化传感器、读取数据等方法。例如,`begin()`函数用于设置传感器的工作模式和I2C地址,`getAcceleration()`和`getGyroscope()`则分别用于获取加速度和角速度数据。 在Arduino项目中,首先需要包含`MPU6050.h`头文件,然后创建`MPU6050`对象,并调用`begin()`函数初始化传感器。之后,可以通过循环调用`getAcceleration()`和`getGyroscope()`来不断更新传感器读数。为了处理这些原始数据,通常还需要进行校准和滤波,以消除噪声和漂移。 I2C通信协议是MPU6050与Arduino交互的基础,它是一种低引脚数的串行通信协议,允许多个设备共享一对数据线。Arduino板上的Wire库提供了I2C通信的底层支持,使得用户无需深入了解通信细节,就能方便地与MPU6050交互。 MPU6050传感器的数据包括加速度(X、Y、Z轴)和角速度(同样为X、Y、Z轴)。加速度数据可以用来计算物体的静态位置和动态运动,而角速度数据则能反映物体转动的速度。结合这两个数据,可以进一步计算出物体的姿态(如角度和角速度变化)。 在嵌入式开发领域,特别是使用STM32微控制器时,也可以找到类似的库来驱动MPU6050。STM32通常具有更强大的处理能力和更多的GPIO口,可以实现更复杂的控制算法。然而,基本的传感器操作流程和数据处理原理与Arduino平台相似。 在实际应用中,除了基本的传感器读取,还可能涉及到温度补偿、低功耗模式设置、DMP(数字运动处理器)功能的利用等高级特性。DMP可以帮助处理传感器数据,实现更高级的运动估计,减轻主控制器的计算负担。 MPU6050是一个强大的六轴传感器,广泛应用于各种需要实时运动追踪的项目中。通过 Arduino 或 STM32 的库文件,开发者可以轻松地与传感器交互,获取并处理数据,实现各种创新应用。博客和其他开源资源是学习和解决问题的重要途径,通过这些资源,开发者可以获得关于MPU6050的详细信息和实践指南
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值