作者:小傅哥 博客:https://bugstack.cn - 包含: java 基础,面经手册,Netty4.x,手写Spring,用Java实现JVM,重学Java设计模式,SpringBoot中间件开发,IDEA插件开发,DDD系统架构项目开发,字节码编程...bugstack 虫洞栈

沉淀、分享、成长,让自己和他人都能有所收获!

一、前言

话我放这,踩过的坑越多头发越少!

说来也是奇怪,只要是学编程的,从初次接触的 Java 到安装 JDK、IDEA、MYSQL, 再到接触 Spring、MyBatis、RPC、MQ,哪怕有时候在浅的坑也会跳进去尝尝鲜,一遍抓着头发,一手点着鼠标也几乎是你的常态。你的键盘里总是有很多被抓碎的头发!

但,哪怕是抓了这么头发,还是遇到了一个满脑子都是骚操作的小伙。“傅哥,我的切面怎么拦截不到?我是照着你的《SpringBoot 中间件设计和开发》专栏写的,你给我看看吧,我都弄了一天了”

接下来我带着大家一起看看什么是快乐星球,他是怎么一顿骚操作让切面拦截不到的!

二、满脑子都是骚操作1. 遇到问题

上周,谢飞机(化名)发过来了自己的手撸的中间件源码,说这代码都没有啥怎么就不能切面呢?

java并发编程实战之基础知识(一个Bug让我发现了)(1)

2. 发现问题

看了几遍源码没发现问题,开始调试,还真它哈拉哨的不进这个切面,接下来;

3. 排查问题

要不是IDEA把 .aj 这货显示成 C 类的图标,可能早就发现问题了。紧接着把这错误类的截图发给了谢飞机,问它你是怎么创建的?他说实话了

三、如何正确使用 Aspect 的 .aj 类

AspectJ,简称 AJ 我自己说的

AspectJ 其实也是 AOP 的一种实现技术,功能类似于拦截器,在集成在 IntelliJ IDEA 开发工具里。在使用 IntelliJ IDEA 编写 AspectJ 代码之前需要本机先安装 AspectJ 工具包。否则你的 .aj 类不能运行,同时IDEA类显示出来的 .aj 类,也是C的标识

接下来我们就来聊聊关于这个东西怎么使用,别再被 .aj 骗了。

1. 安装 AspectJ

在使用 AspectJ 之前,需要去官网下载一个安装包,地址:https://www.eclipse.org/aspectj/downloads.php 如果官网下载的很慢,可以从我提供的源码中获取,也可以从其他途径搜索下载 aspectj-1.9.4.jar

下载完成安装;

java并发编程实战之基础知识(一个Bug让我发现了)(2)

2. AspectJ 插件

在专业版 IDEA 中开发 AspectJ,需要安装以下两个插件:

java并发编程实战之基础知识(一个Bug让我发现了)(3)

3. 添加依赖 aspectjrt.jar

开始之前需要在项目中添加 aspectjrt.jar 依赖,aspectjrt.jar 即 AspectJ 安装目录中lib目录下的jar包。你可以复制到工程中引入,也可以直接引入

java并发编程实战之基础知识(一个Bug让我发现了)(4)

  1. 在工程上鼠标右键,点击 Open Module Setting 打开 Project Structure
  2. 点击 Libraries 选项卡,和上面的 号,创建 New Project Library
  3. 选择 C:\aspectj1.9\lib\aspectjrt.jar 路径,点击即可配置完成
4. 配置AspectJ编译器

IDEA 默认使用 javac 编译器,这里需要配置 AspectJ 的编译器 ajc,在 IDEA 中做相应配置。

java并发编程实战之基础知识(一个Bug让我发现了)(5)

  1. 打开 IDEA -> File -> Settings 对话
  2. 选择 Build,Execution,Deployment -> Compiler -> Java Compiler
  3. Use complier:选择 Ajc
  4. 在 Path to aspectjtools.jar 里配置路径 C:\aspectj1.9\lib\aspectjtools.jar
5. 案例测试

创建 Aspect 类

public aspect DoAspect { pointcut logPointcut():call(* ApiTest.hi(..)); void around():call(void ApiTest.hi(..)){ System.out.println("call 开始..."); proceed(); System.out.println("call 结束..."); } before(): logPointcut(){ System.out.println("方法执行 before"); } after(): logPointcut(){ System.out.println("方法执行 after"); } }

测试类

public class ApiTest { public void hi(){ System.out.println("Hi Aspect"); } public static void main(String[] args) { ApiTest apiTest = new ApiTest(); apiTest.hi(); } }

测试结果

call 开始... 方法执行 before Hi Aspect call 结束... 方法执行 after Process finished with exit code 0

四、总结,