AOP增强:AJc编译技术解析

适合有一定spring基础的人阅读。

ajc编译

此方法是再编译阶段进行的增强,大家先看效果,后面我会讲解原理及使用。

演示

首先来介绍一下大家可能会比较陌生的吧,ajc编译,大家看到这个可能不知道是个什么玩意儿,话不多说,先看看效果。

我们先来定义两个类,用来增强的类和目标类。

目标类:

java 复制代码
@Service // 添加该类的bean到容器中,方便后续的测试,不加也可以增强(后续会将)
public class MyService {
    // 使用日志来打印,方便观察结果,不要使用@Slf4j注解,该增强方法不兼容,编译会报错
    private static final Logger log = LoggerFactory.getLogger(MyService.class);
    
    public void foo() {
        log.info("我是目标方法!");
    }
}

增强类:

java 复制代码
@Aspect// 不用加@Component后续会讲增强原理
public class MyAspect {
    // 作用同上
    private static final Logger log = LoggerFactory.getLogger(MyAspect.class);
    // 通知方法,匹配我上面的目标方法,根据自身情况改变
    @Before("execution(* com.huang.service.MyService.*())")
    public void before() {
        log.info("before");
    }
}

启动类:

java 复制代码
@SpringBootApplication
public class Main {
    public static void main(String[] args) {
        // 获取springboot的容器对象
        ConfigurableApplicationContext run = SpringApplication.run(Main.class, args);
        // 从容器中获得目标bean
        MyService service = run.getBean(MyService.class);
        // 打印查看获取到的bean类型,是原始类型还是代理
        System.out.println(service.getClass());
        // 运行方法,查看是否被增强
        service.foo();
    }
}

运行截图奉上:

从图中我们可以看到,我们的 MyService 类并没有被代理,但是foo()方法被增强了。

其实这个是使用了Aspectj的编译器进行的增强,把我们的MyService的类进行了改写(后续会讲怎么使用的)。

找到项目中MyService的字节码文件打开(idea会自动进行反编译)

可以看到日志输出之前调用before()方法

总结

我们的增强类没有加@Component,并不是由spring进行的管理,而是由Aspectj的编译器来读到这个切面,根据这个切面来进行增强,所以我们不需要使用spring也可以达到增强的效果。

修改后的启动类:

java 复制代码
public class Main {
    public static void main(String[] args) {
        // 直接new对象
        MyService myService = new MyService();
        // 调用方法
        myService.foo();
    }
}

运行结果:

从运行结果中可以看到, MyService类依然被增强了

使用

我们需要在项目中添加一个编译器插件,maven代码如下:

xml 复制代码
<build>
    <plugins>
        <!-- 编译时织入插件 -->
        <plugin>
            <groupId>dev.aspectj</groupId>
            <artifactId>aspectj-maven-plugin</artifactId>
            <version>1.14</version>
            <configuration>
                <complianceLevel>8</complianceLevel> <!-- Java 版本 -->
                <source>8</source>
                <target>8</target>
                <showWeaveInfo>true</showWeaveInfo>
                <forceAjcCompile>true</forceAjcCompile>
            </configuration>
            <executions>
                <execution>
                    <goals>
                        <goal>compile</goal>
                        <goal>test-compile</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

这里我只给出了编译器插件的部分,如果需要使用其他的依赖,请自行添加依赖。

如果添加了之后运行没有被增强,可以先使用maven编译之后再运行

请不要使用@Slf4j注解,Aspectj编译器并不支持

此方法可以增强静态方法,这是代理所不能做到

相关推荐
阿正的梦工坊25 分钟前
【Rust】02-变量、不可变性与基础类型
开发语言·后端·rust
我叫黑大帅2 小时前
通过php 中的Route:: 的写法了解什么是静态类调用
后端·面试·php
JS菌2 小时前
AI Agent 沙箱双层防护体系:从权限过滤到内核隔离的完整实现
前端·人工智能·后端
IT空门:门主3 小时前
Spring 注入三剑客:@Resource、@Autowired、@RequiredArgsConstructor 到底该用哪个?
java·后端·spring
ServBay3 小时前
云端 AI 蜜月期宣告结束,为什么 2026 年开发者转向本地优先架构
后端·ai编程
IT_陈寒3 小时前
Vite这个坑我帮你踩了,动态导入居然这样才生效
前端·人工智能·后端
Sam_Deep_Thinking3 小时前
Spring Boot 的启动原理是什么?
java·spring boot·后端
南部余额3 小时前
Spring WebClient 从入门到精通
java·后端·spring
摇滚侠4 小时前
Spring 零基础入门到进阶 基于注解管理 Bean 38-43
xml·java·后端·spring·intellij-idea
SamDeepThinking4 小时前
我们当年是如何真实落地BFF的?
java·后端·架构