AOP的作用就是在基于OCP在不改变原有系统核心业务代码的基础上动态添加一些扩展功能。通常应用于日志的处理,事务处理,权限处理,缓存处理等等

首先,使用AOP需要添加的依赖有:spring-context、aspectjrt(AOP的实现依托于Aspect框架,版本最好不要使用1.8.5)、aspectjweaver。

SpringAOP的配置有xml和注解两种。其中基于注解的配置又有两种方式:

一、直接在xml文件中

<!-- 在spring-configs.xml中配置对类组件的扫描 -->

<context:component-scan

base-package="com.spring"/>

<!-- 启用AOP注解(自动为目标对象创建代理对象) -->

<aop:aspectj-autoproxy/>

在测试方法中用ClassPathXmlApplicationContext初始化容器对象

二、类中基于注解配置、即全注解形式

  @ComponentScan("com.spring.beans")

  @EnableAspectJAutoProxy

  public class AppConfig {

 }

在测试方法中用AnnotationConfigApplicationContext初始化容器对象

几个概念:

①、切面  具体类,使用@Aspect注解声明

②、连接点:JoinPoint 被拦截的方法

③、切入点:pointcut 拦截的内容,多个连接点集合

④、通知:即执行的动作,切面上某个特定连接点上执行的操作

功能增强 通知五种类型@Before、@AfterReturning、@AfterThrowing、@After以及@Around

它们的执行顺序和try-catch-finally相仿:

@Before前置通知,是 目标对象的业务方法执行之前执行

@AterReturning后置通知,是目标对象的业务方法正常结束才会执行,目标方法抛出异常则不执行,相当于try中的return

@AfterThrowing异常通知,是发生异常执行 发生异常才会执行相当于catch

@After最终通知,是无论有没有异常都会执行 相当于finally

@Around是其他四个通知的综合。

需要注意的是,如果发生异常,顺序是 Before--After---AfterThrowing。发生异常,顺序是:Before--After---AterReturning

  

常用AOP切点表达式有:

bean:匹配指定bean id的方法执行  @Before("bean(orderServiceImpl)") 因为该类已交spring管理,所以默认类名的首字母小写就是该类的bean

within 匹配 某包名下的类中的所有方法 如:@Before("within(com.beans.impl.*)" ) 包名+类名,就像在xml文件中定义bean的时候需要写的class属性

execution:细粒度方式 @Before("execution(* com.beans.impl.*.*(..) )") 即:(返回值类型 包名.类名.方法名(参数列表))

区别:bean和within只能指定到类,不能指定具体方法,而execution可以指定到哪一个类中的具体哪一个方法,更加灵活。

 其中使用的注解有:

接口的实现类用@Service注解

@Aspect标识其为一个AOP横切面对象

切入点@Ponitcut

@Order(数值),规定切面的执行顺序,如果是在xml配置,不依赖注解,在没有写order的情况下,两个切面如果对同一个方法,那么它的执行顺序跟配置文件中切面配置的先后顺序一致。

其中@Pointcut可以为通知的路径设置别名,代码入下:

  1. @Pointcut("execution(* com.beans.impl.*.*(..))")
  2. public void method(){}
  3. @Before("method()")
  4. public void fun(){
  5. 具体扩展的方法实体
  6. }

  

AOP是如何为Bean对象创建代理对象的:cglib代理和jdk动态代理,概括的说,如果代理的是一个没有实现接口的目标对象,那么底层采用的是CGLIB代理,如果这个目标对象实现了某个接口那么底层就会采用JDK动态代理。其中CGLIB底层是代理对象继承了该目标对象,属于父子关系,而JDK动态代理的对象和目标对象实现同一个接口,类似兄弟关系。所以使用jdk代理的时候在getBean方法中,第二个参数 不能给一个实现类的.class,只能使用接口.class

输出日志

如果想要输出是哪个目标对象在调用哪个方法,可以使用JoinPoint作为日志输出,代码如下:

  1. //权限管理
  2. @Aspect
  3. @Service
  4. @Order(1)
  5. public class permissionAspect{
  6. private Logger logger = Logger.getLogger(Permission.Class.getName());
  7.  
  8. @Ponitcut("execution(* com.beans.impl.*.*(..))")
  9. public void checkPermission(JoinPoint point) throws Throwable(){
  10. //JointPoint是连接点对象,此对象中会封装目标对象以及目标方法相关信息,下面这句是:获取连接点所在类的名字
  11. String clsName=point.getTarget().getClass().getName();
  12. //getSignature()该方法是获取方法签名
  13. String methodName=point.getSignature().getName();
  14.  
  15. //打印日志信息
  16. logger.info("类名:"+clsName+“方法名:”+methodName+"权限监测")
  17. }
  18. }

  

spring AOP知识点总结以及日志的输出的更多相关文章

  1. Spring Boot 2.X(八):Spring AOP 实现简单的日志切面

    AOP 1.什么是 AOP ? AOP 的全称为 Aspect Oriented Programming,译为面向切面编程,是通过预编译方式和运行期动态代理实现核心业务逻辑之外的横切行为的统一维护的一 ...

  2. Spring AOP 实现写事件日志功能

    什么是AOP?AOP使用场景?AOP相关概念?Spring AOP组件?如何使用Spring AOP?等等这些问题请参考博文:Spring AOP 实现原理 下面重点介绍如何写事件日志功能,把日志保存 ...

  3. springboot—spring aop 实现系统操作日志记录存储到数据库

    原文:https://www.jianshu.com/p/d0bbdf1974bd 采用方案: 使用spring 的 aop 技术切到自定义注解上,针对不同注解标志进行参数解析,记录日志 缺点是要针对 ...

  4. spring AOP自定义注解 实现日志管理

    今天继续实现AOP,到这里我个人认为是最灵活,可扩展的方式了,就拿日志管理来说,用Spring AOP 自定义注解形式实现日志管理.废话不多说,直接开始!!! 关于配置我还是的再说一遍. 在appli ...

  5. 利用Spring AOP自定义注解解决日志和签名校验

    转载:http://www.cnblogs.com/shipengzhi/articles/2716004.html 一.需解决的问题 部分API有签名参数(signature),Passport首先 ...

  6. Spring AOP 切面实现操作日志

    创建接口注解日志类 package com.fh.service.logAop; /** * Created by caozengling on 2018/7/21. */ import java.l ...

  7. (转)利用Spring AOP自定义注解解决日志和签名校验

    一.需解决的问题 部分API有签名参数(signature),Passport首先对签名进行校验,校验通过才会执行实现方法. 第一种实现方式(Origin):在需要签名校验的接口里写校验的代码,例如: ...

  8. Spring AOP 知识点入门

    一.基本知识点 1.AOP概念 AOP(Aspect-Oriented Programming), 即 面向切面编程, 它与 OOP( Object-Oriented Programming, 面向对 ...

  9. spring aop对service层日志和异常的处理

    1.aop是什么 AOP是Aspect Oriented Programming的缩写,意思是面向切面编程,与OOP(Object Oriented Programming)面向对象编程对等,都是一种 ...

随机推荐

  1. django数据库的表已迁移的不能重新迁移的解决办法

    django.db.utils.InternalError: (1050, "Table 'tb_content' already exists") mysql数据库在迁移时数据库 ...

  2. java-猜数字

    package com.jijy.circle; import java.util.Scanner; import java.util.Random; public class demo5 { pub ...

  3. 关于sql注入漏洞的挖掘及渗透工具简介

    大量的现代企业采用Web应用程序与其客户无缝地连接到一起,但由于不正确的编码,造成了许多安全问题.Web应用程序中的漏洞可使黑客获取对敏感信息(如个人数据.登录信息等)的直接访问. Web应用程序准许 ...

  4. 星星闪烁+多边形移动 canvas

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  5. 查看python内部模块命令,内置函数,查看python已经安装的模块命令

    查看python内部模块命令,内置函数,查看python已经安装的模块命令 可以用dir(modules) 或者用 pip list或者用 help('modules') 或者用 python -m  ...

  6. R语言RODBC数据库操作

    最近准备做一个股票的数据库,用R语言来获取数据,然后存在SQL Sever 2016 express 里面,虽然前面接触过一些操作,但是还是很不熟悉,现在数据已经能获取到了,是时候好好学习一下怎么用R ...

  7. equals方法中变量在前和在后的区别

    对于变量:String str1; 使用str1.equals("null"); 对于变量str1,如果str1是null,空是没有equals方法的,会抛出空指针异常 使用&qu ...

  8. C++算法之大数加法计算的代码

    如下代码段是关于C++算法之大数加法计算的代码,希望对大家有用. { int length; int index; int smaller; int prefix = 0; if(NULL == sr ...

  9. 【leetcode198 解题思路】动态规划

    动态规划 https://blog.csdn.net/so_geili/article/details/53639920 最长公共子序列 https://blog.csdn.net/so_geili/ ...

  10. react native 打包上架

    https://www.jianshu.com/p/ce71b4a8a246 react-native bundle --entry-file index.ios.js --platform ios ...