spring学习笔记(10)@AspectJ研磨分析[1]入门、注解基本介绍

简介: <div class="markdown_views"><h1 id="aspectj准备">@AspectJ准备</h1><blockquote> <p>AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。 <br> 在使用AspectJ之前,我们需要导

@AspectJ准备

AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。
在使用AspectJ之前,我们需要导入aspectJ相应的jar包,可到我的资源页http://download.csdn.net/detail/qwe6112071/9468329 中下载,而如果使用maven则可直接在pom.xml中加入如下代码:

<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjrt</artifactId>
    <version>${org.aspectj-version}</version>
</dependency>
<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.8.1</version>
</dependency>

AspectJ使用示例

前面我们把增强织入到目标对象是通过在IOC容器中配置代理对象工厂实现的,这样做的缺点很明显,如果我需要为大量的类织入相同的增强时,我需要将每个类都配置进去,操作繁琐,除了这种方法,我们还能使用@AspectJ注解,通过在增强的类方法中标注注解的形式来配置增强,同时我们可以通过切点表达式函数来定义切点,这种切点定位的功能是动态而及其强大的。针对前面文章spring学习笔记(6)AOP增强(advice)配置与应用中同样的示例,我们来看看如何用@AspectJ来实现。

1. 配置切面(在原来前置增强的基础上)

    package test.aop;

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

    @Aspect//注解为切面,让IOC容器识别并配置AOP
    public class BeforeAdvice {

        @Before("execution(public String test.aop.Target.speak(Integer))")//定义前置增强并定位切点,定义到test.aop包下的Target对象的public域以String为返回值类型,入参为Integer的speak方法。
        public void before(JoinPoint joinPoint) throws Throwable {//切点入参
            System.out.println("前置日志记录: "
                    + ((Target) joinPoint.getTarget()).getName() + "调用了"
                    + joinPoint.getSignature().getName() + "方法,传入参数为:"
                    + joinPoint.getArgs()[0]);
        }
    }

2. 定义目标对象(还是原来的target)

package test.aop;
//被代理对象
public class Target{

    private static final String name = "zenghao";

    public String speak(Integer age){
        System.out.println("hello I'm " + age + " years old");
        return "I'm return value";
    }
    public static String getName() {
        return name;
    }
}

3. 配置IOC容器

<aop:aspectj-autoproxy /><!-- 使注解@AspectJ生效 -->
    <!-- 配置前置增强  -->
<bean class="test.aop.BeforeAdvice" /><!-- 这里不用声明id,因为我们无须显示引用此bean,只需要注册让IOC容器识别并完成相关的AOP配置 -->
<bean id="target" class="test.aop.Target" /><!-- 注册目标对象,方便在测试时调用 -->

4. 调用测试函数

@Test
public void test(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("classpath:test/aop/aop.xml");
    Target target = (Target) ac.getBean("target");
    target.speak(21);
}

打印结果:
前置日志记录: zenghao调用了speak方法,传入参数为:21
hello I’m 21 years old
相对于我们之前的配置,使用@AspectJ看起来简洁了很多,但这里只是个简单实例,使用@AspectJ的很多强大特性我们会在后面一一提到。

增强注解

下面先分析增强注解的配置格式。括号中为注解的属性,部分属性含义相同就不重复介绍了。

1. 前置增强

`@Before(value,argNames)`

- value:定义切点
- argsName:指定注解所标注的增强方法的参数名,多个参数用都好隔开。

2.后置增强

 `@AfterReturning(value/pointcut,returning,argNames)`

- value/pointcut:定义切点,若同时定义,value会被pointcut覆盖
- returning:将目标对象的方法绑定到增强的方法中

3. 异常增强

`AfterThrowing(value/pointcut,throwing,argNames)`

- throwing:将抛出的异常绑定到增强方法中

4. 最终(final)增强

`After(value,argNames)`

无论是抛出异常还是正常退出,都会执行此增强

5. 引介增强

`@DeclareParents(value,defaultImpl)`

- defaultImpl:默认的接口实现类

关于引介增强的详细分析,可参考本系列后面的一篇博文 《spring学习笔记(12)引介增强详解:定时器实例:无侵入式动态增强类功能》

对于以上的增强,都可以在增强方法中绑定入参org.aspectj.lang.JoinPoint。
在一开始提到的例子中,我们将切面改造如下:

    @Aspect
    public class BeforeAdvice {

        @Before("execution(public String test.aop.Target.speak(Integer))")
        public void before(JoinPoint joinPoint) throws Throwable {
            System.out.println("前置日志记录: "
                    + ((Target) joinPoint.getTarget()).getName() + "调用了"
                    + joinPoint.getSignature().getName() + "方法,传入参数为:"
                    + joinPoint.getArgs()[0]);
                    + 
            System.out.println(joinPoint.getKind());
            System.out.println(joinPoint.getTarget());//获取目标对象
            System.out.println(joinPoint.getThis());//获取目标对象
            System.out.println(joinPoint.getArgs());//获取目标对象方法入参
            System.out.println(joinPoint.getClass());
            System.out.println(joinPoint.getSignature());//获取完整的方法签名
            System.out.println(joinPoint.getSourceLocation());
            System.out.println(joinPoint.getStaticPart());//获取切点定位信息
        }
    }

我们会得到打印信息:
method-execution
test.aop2.UserController@14febdb5
test.aop2.UserController@14febdb5
[Ljava.lang.Object;@3ce4387d
class org.springframework.aop.aspectj.MethodInvocationProceedingJoinPoint
void test.aop2.UserController.login(String)
org.springframework.aop.aspectj.MethodInvocationProceedingJoinPoint$SourceLocationImpl@34a3ef26
execution(void test.aop2.UserController.login(String))

Joinpoint有个子类ProceedingJoinPoint,但它只能在被@Around注解的增强方法中作入参,相对与JoinPoint,ProceedingJoinPoint 新增了两个方法
1. Object org.aspectj.lang.ProceedingJoinPoint.proceed() throws Throwable。作用类似于环绕增强中的org.aopalliance.intercept.Invocation的proceed方法,用于调用原目标对象方法。
2. Object org.aspectj.lang.ProceedingJoinPoint.proceed(Object[] newArgument) throws Throwable此方法和1的区别在于填入了新的入参,即用newArgument代替了原目标方法的入参。

在切点表达式函数中使用通配符

增强注解中的value/pointcut属性都是用于定义切点的,对于切点的定义,我们常用到切点表达式函数,而在表达式函数中,我们可以通过一些通配符来辅助定位连接点。

1. 通配符类型

涉及到的通配符主要有3种:
1. * 匹配任意一个元素,如返回值,某个包,某个类,某个方法等
2. .. 匹配多个元素,表示类时要和*联合使用(如com.yc.controller..*定位controller包下的所有类 ),表示入参时则单独使用(com.yc.controller.UserController.login(..)定位与对应包、类下参数类型数量任意的名为login的方法,这在使用多态特性和不确定入参时常用)
3. + 按类型匹配指定类及其子类

2. 不同的函数对通配符的支持度不同

可以分为如下3类

支持程度 表达式函数
支持所有通配符 execution(),within()
仅支持+通配符 args(),this(),target()
不支持通配符 @args(),@within,@target,@anotation

逻辑运算符

在切点表达式函数的使用中,除了通过通配符外,我们还可以通过一些逻辑运算符来进一步精确而灵活地定位我们要织入增强的连接点。

操作符 说明 实例
&&/and 取交,要求同时满足 within(com.yc.controller..*) && args(com.ye.model.User)定位controller包下以User作入参的方法
/or 取并,满足其一即可 如上例取并定位controller包下的所有类方法和以User作入参(可以是其他包下)的方法
!/not 取非 within(com.yc.controller..*) && ! args(com.ye.model.User)定位controller包下不User作入参的方法
目录
相关文章
|
8天前
|
Java Spring
【Spring】方法注解@Bean,配置类扫描路径
@Bean方法注解,如何在同一个类下面定义多个Bean对象,配置扫描路径
135 73
|
3天前
|
Java Spring 容器
【SpringFramework】Spring IoC-基于注解的实现
本文主要记录基于Spring注解实现IoC容器和DI相关知识。
36 21
|
8天前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
|
8天前
|
Java Spring
【Spring配置】idea编码格式导致注解汉字无法保存
问题一:对于同一个项目,我们在使用idea的过程中,使用汉字注解完后,再打开该项目,汉字变成乱码问题二:本来a项目中,汉字注解调试好了,没有乱码了,但是创建出来的新的项目,写的注解又成乱码了。
|
4月前
|
SQL 监控 druid
springboot-druid数据源的配置方式及配置后台监控-自定义和导入stater(推荐-简单方便使用)两种方式配置druid数据源
这篇文章介绍了如何在Spring Boot项目中配置和监控Druid数据源,包括自定义配置和使用Spring Boot Starter两种方法。
|
3月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
254 2
|
10天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
|
17天前
|
NoSQL Java Redis
Spring Boot 自动配置机制:从原理到自定义
Spring Boot 的自动配置机制通过 `spring.factories` 文件和 `@EnableAutoConfiguration` 注解,根据类路径中的依赖和条件注解自动配置所需的 Bean,大大简化了开发过程。本文深入探讨了自动配置的原理、条件化配置、自定义自动配置以及实际应用案例,帮助开发者更好地理解和利用这一强大特性。
67 14
|
2月前
|
缓存 IDE Java
SpringBoot入门(7)- 配置热部署devtools工具
SpringBoot入门(7)- 配置热部署devtools工具
56 1
SpringBoot入门(7)- 配置热部署devtools工具
|
2月前
|
缓存 IDE Java
SpringBoot入门(7)- 配置热部署devtools工具
SpringBoot入门(7)- 配置热部署devtools工具
55 2
 SpringBoot入门(7)- 配置热部署devtools工具