AspectJ语法

 AspectJ 通过连接点向Java添加一些新的程序元素来扩展Java,是Java面向切点一种实现。其主要包括连接点、切点、建议、切面及类型间声明。

连接点:程序在执行过程中明确的点。包括方法和构造函数调用及字段访问等。

切点:用来挑选连接点。

建议:在程序运行到由切点挑选的连接点时执行的程序片段。

切面:类似于类,成员包含了连接点、切点及切面等。

类型间声明:静态改变类的结构及层级关系。 可以为已有的类定义新的成员、方法。也可以使其继承新的类及实现接口。

1 AspectJ 语法

aspectJ 需要使用ajc编译器进行编译。

1.1 切点

切点类似于类中的方法,有访问权限修饰符、可以被定义为final,但也有些不同:1)不能重载。2)作用域包括切面声明及主体(方法的作用域是在类主体)。

if

条件判断

within

witin(全限定名),在某个类的全部连接点。

cflow

cflow(pointcut), 连接点pointcut 及之后的且在其封闭区间内的所有连接点。包括在在该方法中所调用的其他方法的连接点。

get

当某个字段被访问时会被捕获。(不能捕获参数)

set

当某个字段被赋值时会被捕获。(可以捕获参数)

handler

try..catch中的catch捕获的错误类型是Throwable类型及其子类型时不被捕获。(可以捕获参数)

adviceexecution

adviceexecution() 所有建议在执行时会被捕获。

表 切点的部分语法

public class PointcutEntity {private String name;public void fun(String val) {this.name = val;fun();System.out.println("-----");}private void fun() {System.out.println(this.name);}public void fun2() {System.out.println("fun2");try {throw new Throwable();} catch (Throwable e) {System.out.println("处理");}}
}public aspect PointcutEntityAspect perthis(withPointcut()){ // 相当于全局声明了,在这个切面中,所有的连接点都在PointcutEntity里运行final pointcut withPointcut(): within(service.PointcutEntity); // PointcutEntity内的所有连接点pointcut funPointcut(): execution(* service.PointcutEntity.fun());pointcut cFlowPointcut(): cflow(funPointcut()); // funPointcut 连接点及之后切在封闭区间内的所以连接点pointcut getPointcut(): get(String service.PointcutEntity.name); // 捕获PointcutEntity的name字段被访问pointcut setPointcut(String name): set(String service.PointcutEntity.name) && args(name); // 捕获PointcutEntity的name字段被赋值pointcut handlerPointcut(Throwable e): handler(Throwable) && args(e); // try..catch中的catch捕获的错误类型是Throwable类型及其子类型时不被捕获//    before(): cFlowPointcut() {
//        System.out.println("cFlowPointcut---");
//        System.out.println(thisJoinPoint.getSourceLocation());
//        System.out.println(thisJoinPoint.getSignature());
//        System.out.println(thisJoinPoint.getKind());
//        System.out.println();
//    }//    before(): getPointcut() {
//        System.out.println("getPointcut---");
//        System.out.println(thisJoinPoint.getSourceLocation());
//        System.out.println(thisJoinPoint.getSignature());
//        System.out.println(thisJoinPoint.getKind());
//        System.out.println();
//    }//    before(String name): setPointcut(name) {
//        System.out.println("getPointcut---");
//        System.out.println("赋值value:" + name);
//        System.out.println(thisJoinPoint.getSourceLocation());
//        System.out.println(thisJoinPoint.getSignature());
//        System.out.println(thisJoinPoint.getKind());
//        System.out.println();
//    }before(Throwable e): handlerPointcut(e) {System.out.println("handlerPointcut---");System.out.println("抛错:" + e);System.out.println(thisJoinPoint.getSourceLocation());System.out.println(thisJoinPoint.getSignature());System.out.println(thisJoinPoint.getKind());System.out.println();}public static void main(String[] args) {PointcutEntity pointcutEntity = new PointcutEntity();pointcutEntity.fun2();}
}

1.2 建议

有三种类型,before、after(包括 after() returning、after() throwing及after)及around。around 可以替换方法的返回值。

public class AdviceEntity {public String fun() {System.out.println("fun");return "建议";}public void fun1() throws ClassNotFoundException {System.out.println("fun1");throw new ClassNotFoundException();}public String fun2() {System.out.println("fun2");return "fun2";}
}public aspect AdviceEntityAspect {pointcut funPointcut(): execution(* service.AdviceEntity.fun*(..));pointcut fun2Pointcut(): execution(* service.AdviceEntity.fun2(..));after() returning (String val): funPointcut() {System.out.println("after() returning");System.out.println("返回值:" + val);System.out.println(thisJoinPoint.getSourceLocation());System.out.println(thisJoinPoint.getSignature());System.out.println();}after() throwing (ClassNotFoundException e): funPointcut() {System.out.println("after() throwing");System.out.println("捕获错误:" + e);System.out.println(thisJoinPoint.getSourceLocation());System.out.println(thisJoinPoint.getSignature());System.out.println();}String around(): fun2Pointcut() {System.out.println("String around()");System.out.println(thisJoinPoint.getSourceLocation());System.out.println(thisJoinPoint.getSignature());System.out.println();return "around() String";}public static void main(String[] args) throws ClassNotFoundException {AdviceEntity entity = new AdviceEntity();entity.fun();System.out.println("fun2的返回值:" + entity.fun2());entity.fun1();}
}

1.3 类型间声明

AspectJ 可以为类定义方法及字段,可以使其继承新的类及实现接口。

public class InterEntity {private final String name = "inter_entity";public void fun() {System.out.println("InterEntity定义的name:" + this.name);this.speak(); // 在类中可以直接使用在切面中定义的方法}
}public interface SpeakInterface {void speak();
}public aspect InterEntityAspect {private String InterEntity.name = "InterEntityAspect"; // 给AdviceEntity定义name字段,虽然这个字段在原结构已存在,// 但是因为这里和AdviceEntity中其访问类型是private,所以不会冲突declare parents: InterEntity implements SpeakInterface; // 定义实现接口,还需要定义实现该接口的方法public void InterEntity.speak() { // 上面接口方法的实现System.out.println("InterEntityAspect ");}private void InterEntity.show() {// 这里的this 是指InterEntity的实例System.out.println(this.name); // InterEntityAspectthis.fun(); // InterEntity定义的name:inter_entitythis.speak();}public static void main(String[] args) {InterEntity interEntity = new InterEntity();interEntity.show();}
}

declare soft: Type: Pointcut; 捕获类型为Type的异常并且重抛成SoftException。(Type 不能是RuntimeException)。

declare precedence: TypePatternList; 为切面定义优先级,TypePatternList为切面列表,“,”隔开,越前面优先级越高。

public class DeclareEntity {public void fun() {System.out.println("fun");}public void fun(int num) throws CustomException {if (num == 0) throw new CustomException();System.out.println(33 / num);}
}public abstract aspect DeclareEntityParentAspect pertarget(DeclareEntityClassPointcut()){pointcut DeclareEntityClassPointcut(): within(service.DeclareEntity);pointcut funPointCut(): execution(* fun(..));declare precedence: DeclareEntityChildrenAspectB,DeclareEntityChildrenAspectA; // 定义切面的优先级declare soft:CustomException: funPointCut();public static void main(String[] args) throws CustomException {DeclareEntity entity = new DeclareEntity();entity.fun();entity.fun(0);}
}public aspect DeclareEntityChildrenAspectA extends DeclareEntityParentAspect{before(): funPointCut() {System.out.println("DeclareEntityChildrenAspectA");System.out.println(thisJoinPoint.getSourceLocation());System.out.println();}
}public aspect DeclareEntityChildrenAspectB extends DeclareEntityParentAspect{before(): funPointCut() {System.out.println("DeclareEntityChildrenAspectB");System.out.println(thisJoinPoint.getSourceLocation());System.out.println();}
}

1.4 切面

切面像类一样也可以被继承,被继承的切面须为抽象切面。 也可以继承类及实现接口。

[issingleton()]

默认类型,将不限定切点。

perthis(Pointcut)

全局限定连接点在this(Pointcut)中。

pertarget(Pointcut)

全局限定连接点在target(Pointcut)中。

percflow(Pointcut)

全局限定连接点在cflow(Pointcut)z中。

percflowbelow(Pointcut)

全局限定连接点在cflowbelow(Pointcut)中。

图 五种在切面中全局限定连接点的方式

2 AspectJ 的应用场景

2.1 开发阶段

我们在开发过程中,需要进行调试,有时会往业务代码中插入跟踪代码。在上线的时候我们要把这些代码删除,否则可能会降低系统性能。有时我们插入的代码会比较多和广,可能会漏删这些代码。因为有些人常常会通过写脚本的方式来进行调试。

而AspectJ 在这方面具有独特的优势,可以实现对调试代码的“插拔”。

需求:打印develop.service 包下所有类以get或set开头方法访问日志。

public class Student {private String name;private Integer age;public String getName() {return name;}public void setName(String name) {this.name = name;}public Integer getAge() {return age;}public void setAge(Integer age) {this.age = age;}
}public class Teacher {private String name;private String course;public String getName() {return name;}public void setName(String name) {this.name = name;}public String getCourse() {return course;}public void setCourse(String course) {this.course = course;}
}public class GetAndSetTrace {public static void info(String className,String methName,String location,Object result,Object... args) {String sb = "访问时间:" + new Date() + "\n" +"类名:" + className + "\n" +"方法:" + methName + "\n";if (args != null) sb += "参数名:" + Arrays.asList(args) + "\n";if (result != null) sb +="返回值:" + result + "\n";sb += "连接点:" + location;System.out.println(sb);System.out.println("--------------------");}
}public aspect GetAndSetTraceAspect pertarget(targetClassPointcut()){pointcut targetClassPointcut(): within(develop.service.*);pointcut getPointcut(): execution(Object+ get*());pointcut setPointcut(Object obj): execution(void set*(*)) && args(obj);after() returning(Object obj): getPointcut() {trace(thisJoinPoint,obj);}before(Object obj): setPointcut(obj) {trace(thisJoinPoint,null);}private void trace(JoinPoint joinPoint,Object result) {String className = joinPoint.getTarget().getClass().getName();String methodName = joinPoint.getSignature().toShortString();String location = joinPoint.getSourceLocation().getLine() + "";GetAndSetTrace.info(className,methodName,location,result,joinPoint.getArgs());}public static void main(String[] args) {Student student = new Student();Teacher teacher = new Teacher();student.setName("小名同学");student.setAge(18);teacher.setName("刘老师");teacher.setCourse("英语");System.out.println("学生:" + student.getName());System.out.println("年龄:" + student.getAge());System.out.println("教师:" + teacher.getName());System.out.println("课程:" + teacher.getCourse());}
}

AspectJ 可重用,比如上面需求有改动,对日志打印格式及要求传递的参数做了修改。如果不使用Aspect而使用传统方式-往业务代码插入跟踪代码。那么每个被插入的跟踪代码都需要修改。而对于AspectJ 就仅仅修改相关切面及跟踪类即可。

2.2 生产阶段

观察者模式有两种角色:Subject目标类,被观察的对象;Observer 观察者,将观察的目标的改变而做出改变。

PropertyChangeSupport和PropertyChangeListener类是Java中用来监听对象属性变化的。

public class User {private String name;private final PropertyChangeSupport propertyChangeSupport = new PropertyChangeSupport(this);public void addPropertyChangeListener(PropertyChangeListener listener) {propertyChangeSupport.addPropertyChangeListener(listener);}public void setName(String name) {propertyChangeSupport.firePropertyChange("name",this.name,name);this.name = name;}public String getName() {return name;}public static void main(String[] args) {User user = new User();PropertyChangeListener listener = new PropertyChangeListener() {@Overridepublic void propertyChange(PropertyChangeEvent evt) {System.out.println(evt.getPropertyName() + "发生了改变,旧值:" + evt.getOldValue() + ",新值:" + evt.getNewValue());}}; // 监听器user.addPropertyChangeListener(listener);user.setName("黄先生");user.setName("刘女士");}
}

我们可以使用AspectJ 实现上面的功能。

public aspect PropertyChangeAspect{private Set<PropertyChangeObserver> User.observerSet = new HashSet<>();public void User.addListener(PropertyChangeObserver observer) {observerSet.add(observer);}public Set<PropertyChangeObserver> User.getListenerSet() {return this.observerSet;}pointcut setMethod(Object arg): execution(* set*(*)) && args(arg);before(Object org): setMethod(org) {Object target = thisJoinPoint.getTarget();if (target instanceof User) {for(PropertyChangeObserver listener : ((User) target).getListenerSet()) {listener.update(thisJoinPoint.getSignature().getName(),org);}}System.out.println();}public static void main(String[] args) {User user = new User();PropertyChangeObserver observer1 = (methodName, arg) -> {System.out.println("观察者1,方法:" + methodName + ",参数:" + arg);};PropertyChangeObserver observer2 = new PropertyChangeObserver() {@Overridepublic void update(String methodName, Object arg) {System.out.println("观察者2,参数:" + arg);}};user.addListener(observer1);user.addListener(observer2);user.setName("黄");user.setName("刘");}// 自定义建设者interface PropertyChangeObserver  {void update(String methodName,Object arg);}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/632172.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ubuntu开放ssh服务

&#x1f4d1;前言 本文主要是【ubuntu】——ubuntu开放ssh服务的文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是听风与他&#x1f947; ☁️博客首页&#xff1a;CSDN主页听风与他 &#x1f304;每日一…

【安装VMware Tools】实现Vmware虚拟机和主机之间复制、粘贴内容、拖拽文件

https://www.bilibili.com/video/BV1rN411277B/?spm_id_from333.788.recommend_more_video.6&vd_sourcefb8dcae0aee3f1aab700c21099045395 https://blog.csdn.net/wxqian25/article/details/19406673 待解决方案&#xff1a; 重新下载ubuntu&#xff0c;然后按照 https://…

C++:通过ofstream写入二进制文件内容

C++:通过ifstream读取二进制文件内容_c++ ifstream 二进制读取-CSDN博客 介绍了读取二进制文件的方法。 本文介绍一下写入二进制数据到文件的方法: 1.通过write #include <fstream> #include <string> using namespace std; int main() {int data = 0x0102030…

01 SpringMVC的快速理解

1.1 如图所示&#xff0c;SpringMVC负责表述层&#xff08;控制层Controller&#xff09;实现简化&#xff01; SpringMVC的作用主要覆盖的是表述层&#xff0c;例如&#xff1a; 请求映射、数据输入、视图界面、请求分发、表单回显、会话控制、过滤拦截、异步交互、文件上传…

【协议】XMLHttpRequest的梳理和总结

1. 前言 本篇梳理和总结一下XMLHttpRequest。 2. XMLHttpRequest原型对象的属性和方法 属性和方法说明示例new XMLHttpRequest() 功能&#xff1a;创建XHR对象 输入&#xff1a; 输出&#xff1a;XHR实例化对象 <略> XMLHttpRequest.prototype .open(method, url, asyn…

CDH6.3.2企业级安装实战

一、系统准备 1、环境介绍 IP操作系统联网10.191.15.15Centos 7.4离网10.191.15.16Centos 7.4离网10.191.15.17Centos 7.4离网10.191.15.18Centos 7.4离网2、搭建本地Yum源 2.1 配置本地基础Yum源 1、上传镜像到服务器 下载的Centos镜像为 CentOS-7-x86_64-Everything-1708…

STM32存储左右互搏 SPI总线FATS读写FRAM MB85RS2M

STM32存储左右互搏 SPI总线FATS读写FRAM MB85RS2M 在中低容量存储领域&#xff0c;除了FLASH的使用&#xff0c;&#xff0c;还有铁电存储器FRAM的使用&#xff0c;相对于FLASH&#xff0c;FRAM写操作时不需要预擦除&#xff0c;所以执行写操作时可以达到更高的速度&#xff0…

渗透测试之Hydra如何B破远程主机RDP登录M码

环境: Hydra9.3 KALI2022 问题描述: 渗透测试之hydra如何B破远程主机RDP登录M码 解决方案: Hydra是一款非常强大的网络登录P解工具。它专门用于测试和评估网络安全,通过暴力P解方式尝试多种用户名和密码组合,以获得对受测试系统的非法访问。Hydra支持各种协议的登录破…

一款 StarRocks 客户端工具,支持可视化建表、数据编辑

什么是 StarRocks&#xff1f; StarRocks 是新一代极速全场景 MPP (Massively Parallel Processing) 数据库。StarRocks 的愿景是能够让用户的数据分析变得更加简单和敏捷。用户无需经过复杂的预处理&#xff0c;就可以用 StarRocks 来支持多种数据分析场景的极速分析。 为了…

Oracle行转列函数,列转行函数

Oracle行转列函数&#xff0c;列转行函数 Oracle 可以通过PIVOT,UNPIVOT,分解一行里面的值为多个列,及来合并多个列为一行。 PIVOT PIVOT是用于将行数据转换为列数据的查询操作(类似数据透视表)。通过使用PIVOT&#xff0c;您可以按照特定的列值将数据进行汇总&#xff0c;并将…

vue中父组件异步传值,渲染问题

vue中父组件异步传值&#xff0c;渲染问题 父组件异步传值&#xff0c;子组件渲染不出来。有如下两种解决方法&#xff1a; 1、用v-if解决&#xff0c;当父组件有数据才渲染 <Child v-if"dataList && dataList.length > 0" :data-list"dataLis…

C# 使用Bitmap 将byte[] 转成.jpg/.png/gif等图片

在 C# 中&#xff0c;你可以使用 System.Drawing 命名空间中的 Bitmap 类来将 byte[] 转换为 .jpg 图片。以下是一个示例代码&#xff1a; using System; using System.Drawing; using System.IO;class Program {static void Main(){byte[] imageBytes GetImageBytes(); // 获…

第三届iEnglish全国ETP大赛展现教育游戏新趋势

随着社会步入数字化纪元,游戏作为信息交流和传播的重要载体,在教育领域的潜能日益凸显。特别是寓教于乐的“教育游戏”学习方式让更多家长和孩子体验到“玩中学,学中玩”的乐趣,在教育领域的潜能也日益凸显。 本周五(1月19日)晚上7点,国内首个教育游戏赛事、以“玩转英语,用iE…

光伏电站整体解决方案:光伏开发、设计和施工一体化

鹧鸪云 光伏市场前景、未来趋势 光伏市场前景广阔&#xff0c;政府对于新能源的支持力度在不断加大&#xff0c;市场规模也随之扩大&#xff1b;光伏技术不断创新&#xff0c;致使光伏发电成本降低&#xff0c;效率提高&#xff1b;随着光伏组件成本的降低和安装技术的成熟&a…

2024年阿里云服务器地域所在城市分布表

2024年阿里云服务器地域分布表&#xff0c;地域指数据中心所在的地理区域&#xff0c;通常按照数据中心所在的城市划分&#xff0c;例如华北2&#xff08;北京&#xff09;地域表示数据中心所在的城市是北京。阿里云地域分为四部分即中国、亚太其他国家、欧洲与美洲和中东&…

py爬虫入门笔记(request.get的使用)

文章目录 Day11. 了解浏览器开发者工具2. Get请求http://baidu.com3. Post请求https://fanyi.baidu.com/sug4. 肯德基小作业 Day21. 正则表达式2. 使用re模块3. 爬取豆瓣电影Top250的第一页4. 爬取豆瓣电影Top250所有的250部电影信息 Day31. xpath的使用2. 认识下载照片线程池的…

java多线程(线程池)

1、创建一个可缓存线程池&#xff0c;如果线程池长度超过处理需要&#xff0c;可灵活回收空闲线程&#xff0c;若无可回收&#xff0c;则新建线程。 public static void main(String[] args) {ExecutorService cachedThreadPool Executors.newCachedThreadPool();for (int i …

IOS-相机权限申请-Swift

配置描述 在Info.plist文件中&#xff0c;新建一个键值对Privacy - Camera Usage Description&#xff08;或者NSCameraUsageDescription&#xff09;&#xff0c;值为申请描述说明&#xff0c;自定义的 申请 然后在需要申请的文件中导入AVFoundation import AVFoundation…

Python 常用模块struct

【一】介绍 struct 模块提供了一种处理二进制数据的方式它允许你使用类似于C语言的结构体的方式来打包&#xff08;pack&#xff09;和解包&#xff08;unpack&#xff09;数据这对于处理二进制文件、网络协议等场景非常有用。 【二】使用 【1】打包 &#xff08;1&#xf…

CSS||Emmet语法

1、简介 ​ Emmet语法的前身是Zen coding,它使用缩写,来提高html/css的编写速度, Vscode内部已经集成该语法。 ​ 快速生成HTML结构语法 ​ 快速生成CSS样式语法 2、快速生成HTML结构语法 生成标签 直接输入标签名 按tab键即可 比如 div 然后tab 键&#xff0c; 就可以生成 <…