AspectJ的基本用法
发布时间 - 2026-01-11 00:33:00 点击率:次AOP虽然是方法论,但就好像OOP中的Java一样,一些先行者也开发了一套语言来支持AOP。目前用得比较火的就是AspectJ了,它是一种几乎和Java完全一样的语言,而且完全兼容Java(AspectJ应该就是一种扩展Java,但它不是像Groovy[1]那样的拓展。)。当然,除了使用AspectJ特殊的语言外,AspectJ还支持原生的Java,只要加上对应的AspectJ注解就好。所以,使用AspectJ有两种方法:

完全使用AspectJ的语言。这语言一点也不难,和Java几乎一样,也能在AspectJ中调用Java的任何类库。AspectJ只是多了一些关键词罢了。
或者使用纯Java语言开发,然后使用AspectJ注解,简称*@AspectJ*。
AspectJ的配置可以参考另一篇文章Android中使用AspectJ详解
Join Points介绍
Join Points是AspectJ中的一个关键概念。Join Points可以看作是程序运行时的一个执行点,比如:一个函数的调用可以看作是个Join Points,如Log.e()这个函数,e()可以看作是个Join Points,而调运e()的函数也可以认为是一个Join Points;设置一个变量,或者读取一个变量也可以是个Join Points;for循环也可以看作是Join Points。
理论上说,一个程序中很多地方都可以被看做是Join Points,但是AspectJ中,只有下面所示的几种执行点被认为是Join Points:
| Join Points | 说明 | 示例 |
|---|---|---|
| method call | 函数调用 | 比如调用Log.e(),这是一处JPoint |
| method execution | 函数执行 | 比如Log.e()的执行内部,是一处Join Points。注意它和method call的区别。method call是调用某个函数的地方。而execution是某个函数执行的内部。 |
| constructor call | 构造函数调用 | 和method call类似 |
| constructor execution | 构造函数执行 | 和method execution类似 |
| field get | 获取某个变量 | 比如读取DemoActivity.debug成员 |
| field set | 设置某个变量 | 比如设置DemoActivity.debug成员 |
| pre-initialization | Object在构造函数中做得一些工作。 | |
| initialization | Object在构造函数中做得工作 | |
| static initialization | 类初始化 | 比如类的static{} |
| handler | 异常处理 | 比如try catch(xxx)中,对应catch内的执行 |
| advice execution | 这个是AspectJ的内容,稍后再说 |
这里列出了AspectJ所认可的JoinPoints的类型。实际上,也就是你想把新的代码插在程序的哪个地方,是插在构造方法中,还是插在某个方法调用前,或者是插在某个方法中,这个地方就是Join Points,当然,不是所有地方都能给你插的,只有能插的地方,才叫Join Points。
Pointcuts介绍
一个程序会有多个Join Points,即使同一个函数,也还分为call和execution类型的Join Points,但并不是所有的Join Points都是我们关心的,Pointcuts就是提供一种使得开发者能够选择自己需要的JoinPoints的方法。
Advice
Advice就是我们插入的代码以何种方式插入,有Before还有After、Around。
看个例子
@Before("execution(* android.app.Activity.on**(..))")
public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable {
}
这里会分成几个部分,我们依次来看:
- @Before:Advice,也就是具体的插入点
- execution:处理Join Point的类型,例如call、execution
- (* android.app.Activity.on**(..)):这个是最重要的表达式,第一个*表示返回值,*表示返回值为任意类型,后面这个就是典型的包名路径,其中可以包含 * 来进行通配,几个 * 没区别。同时,这里可以通过&&、||、!来进行条件组合。()代表这个方法的参数,你可以指定类型,例如android.os.Bundle,或者(..)这样来代表任意类型、任意个数的参数。
- public void onActivityMethodBefore:实际切入的代码。
Before和After其实还是很好理解的,也就是在Pointcuts之前和之后,插入代码,那么Around呢,从字面含义上来讲,也就是在方法前后各插入代码,是的,他包含了Before和After的全部功能,代码如下:
@Around("execution(* com.xys.aspectjxdemo.MainActivity.testAOP())")
public void onActivityMethodAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
String key = proceedingJoinPoint.getSignature().toString();
Log.d(TAG, "onActivityMethodAroundFirst: " + key);
proceedingJoinPoint.proceed();
Log.d(TAG, "onActivityMethodAroundSecond: " + key);
}
其中,proceedingJoinPoint.proceed()代表执行原始的方法,在这之前、之后,都可以进行各种逻辑处理。
自定义Pointcuts
自定义Pointcuts可以让我们更加精确的切入一个或多个指定的切入点。
首先我们要定义一个注解类
@Retention(RetentionPolicy.CLASS)
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD})
public @interface DebugTrace {
}
在需要插入代码的地方加入这个注解。如在MainActivity中加入,
public class MainActivity extends AppCompatActivity {
final String TAG = MainActivity.class.getSimpleName();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
logTest();
}
@DebugTrace
public void logTest() {
Log.e(TAG, "log test");
}
}
最后,创建切入代码
@Pointcut("execution(@com.kun.aspectjtest.aspect.DebugTrace * *..*.*(..))")
public void DebugTraceMethod() {}
@Before("DebugTraceMethod()")
public void beforeDebugTraceMethod(JoinPoint joinPoint) throws Throwable {
String key = joinPoint.getSignature().toString();
Log.e(TAG, "beforeDebugTraceMethod: " + key);
}
log如下
在AspectJ的切入点表达式中,我们前面都是使用的execution,实际上,还有一种类型——call,那么这两种语法有什么区别呢,对于Call来说:
Call(Before)
Pointcut{
Pointcut Method
}
Call(After)
对于Execution来说:
Pointcut{
execution(Before)
Pointcut Method
execution(After)
}
withincode
这个语法通常来进行一些切入点条件的过滤,作更加精确的切入控制。如下
public class MainActivity extends AppCompatActivity {
final String TAG = MainActivity.class.getSimpleName();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
aspectJ1();
aspectJ2();
aspectJ3();
}
public void aspectJTest() {
Log.e(TAG, "execute aspectJTest");
}
public void aspectJ1(){
aspectJTest();
}
public void aspectJ2(){
aspectJTest();
}
public void aspectJ3(){
aspectJTest();
}
}
aspectJ1(),aspectJ2(),aspectJ3()都调用了aspectJTest方法,但只想在aspectJ2调用aspectJTest时插入代码,这个时候就需要使用到Pointcut和withincode组合的方式,来精确定位切入点。
@Pointcut("(call(* *..aspectJTest()))&&withincode(* *..aspectJ2())")
public void invokeAspectJTestInAspectJ2() {
}
@Before("invokeAspectJTestInAspectJ2()")
public void beforeInvokeaspectJTestInAspectJ2(JoinPoint joinPoint) throws Throwable {
Log.e(TAG, "method:" + getMethodName(joinPoint).getName());
}
private MethodSignature getMethodName(JoinPoint joinPoint) {
if (joinPoint == null) return null;
return (MethodSignature) joinPoint.getSignature();
}
log如下
04-02 23:44:40.681 12107-12107/ E/MainActivity: execute aspectJTest 04-02 23:44:40.681 12107-12107/ E/AspectTest: method:aspectJTest 04-02 23:44:40.681 12107-12107/ E/MainActivity: execute aspectJTest 04-02 23:44:40.681 12107-12107/ E/MainActivity: execute aspectJTest
以上就是Aspecj的基本使用方法,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!
# aspectj
# 在Android项目中使用AspectJ的方法
# AndroidStudio 配置 AspectJ 环境实现AOP的方法
# Spring Aop之AspectJ注解配置实现日志管理的方法
# Android AOP框架AspectJ使用详解
# 分析java 中AspectJ切面执行两次的原因
# Android中使用AspectJ详解
# 详解Spring Aop实例之AspectJ注解配置
# Spring使用AspectJ注解和XML配置实现AOP
# java基于AspectJ(面向切面编程)编码示例分享
# Spring AOP AspectJ使用及配置过程解析
# 关键词
# 是个
# 插在
# 都是
# 几个
# 是在
# 也就
# 多个
# 做得
# 自定义
# 是一个
# 这是
# 很好
# 会有
# 给你
# 出了
# 你可以
# 第一个
# 让我们
# 在这
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐
laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法
手机软键盘弹出时影响布局的解决方法
Javascript中的事件循环是如何工作的_如何利用Javascript事件循环优化异步代码?
Laravel怎么集成Vue.js_Laravel Mix配置Vue开发环境
如何快速重置建站主机并恢复默认配置?
Laravel Docker环境搭建教程_Laravel Sail使用指南
Laravel怎么发送邮件_Laravel Mail类SMTP配置教程
Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧
新三国志曹操传主线渭水交兵攻略
Laravel如何处理文件上传_Laravel Storage门面实现文件存储与管理
Java类加载基本过程详细介绍
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能
php 三元运算符实例详细介绍
HTML5空格和nbsp有啥关系_nbsp的作用及使用场景【说明】
如何快速搭建高效服务器建站系统?
UC浏览器如何设置启动页 UC浏览器启动页设置方法
魔毅自助建站系统:模板定制与SEO优化一键生成指南
Laravel事件和监听器如何实现_Laravel Events & Listeners解耦应用的实战教程
Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤
PHP的CURL方法curl_setopt()函数案例介绍(抓取网页,POST数据)
制作电商网页,电商供应链怎么做?
如何快速上传自定义模板至建站之星?
Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】
Laravel如何从数据库删除数据_Laravel destroy和delete方法区别
Laravel怎么生成二维码图片_Laravel集成Simple-QrCode扩展包与参数设置【实战】
javascript中的try catch异常捕获机制用法分析
微信小程序 五星评分(包括半颗星评分)实例代码
ai格式如何转html_将AI设计稿转换为HTML页面流程【页面】
网站优化排名时,需要考虑哪些问题呢?
如何在IIS中新建站点并解决端口绑定冲突?
JS去除重复并统计数量的实现方法
什么是javascript作用域_全局和局部作用域有什么区别?
Laravel Session怎么存储_Laravel Session驱动配置详解
如何确保西部建站助手FTP传输的安全性?
Laravel用户密码怎么加密_Laravel Hash门面使用教程
Laravel如何使用Socialite实现第三方登录?(微信/GitHub示例)
宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程
JavaScript如何操作视频_媒体API怎么控制播放
b2c电商网站制作流程,b2c水平综合的电商平台?
Mybatis 中的insertOrUpdate操作
Bootstrap整体框架之JavaScript插件架构
免费视频制作网站,更新又快又好的免费电影网站?
Laravel如何处理文件下载请求?(Response示例)
谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程
Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】
如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?
如何在宝塔面板中修改默认建站目录?
*服务器网站为何频现安全漏洞?
高性能网站服务器部署指南:稳定运行与安全配置优化方案

