注解和aop的简记
创始人
2024-05-31 03:47:25
0

注解和aop的简记

一、注解

常见的注解有如下

标准注解:

  • @Override : 表示当前方法将覆盖超类的方法
  • @Deprecated : 标记过时,如果使用了该注解标记的方法,将会报警。
  • @SuppressWarnings : 关闭警告信息。

1. 注解的释义

注解(也被称为元数据)为我们在代码中添加信息提供了一种形式化的方法,使得我们可以在稍后某个时候非常方便地使用这些数据。

引自 Think in java。

2. 注解的使用

想要定义一个注解,需要元注解的帮助。

元注解:

  • @Target : 表示注解可以用到什么地方
  • @Retention : 表示需要在什么级别保存注解信息
  • @Documented :将注解包含在Javadoc中
  • @Inherited : 允许子类继承父类的注解

简单样例:

/*** 添加这个注解,可以统计方法的运行时间,* 且携带开发者相关信息* @author cay* @since 2023/02/07*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RunningTime {/*** 添加开发负责人的信息,方便对应人员修复bug* @return 开发者相关信息*/String authorMsg();
}

@Target

从上面的简单案例中可以看到 需要填加内容 ElementType, ElementType的是一个枚举类型。

public enum ElementType {/** Class, interface (including annotation type), or enum declaration */TYPE,/** Field declaration (includes enum constants) */FIELD,/** Method declaration */METHOD,/** Formal parameter declaration */PARAMETER,/** Constructor declaration */CONSTRUCTOR,/** Local variable declaration */LOCAL_VARIABLE,/** Annotation type declaration */ANNOTATION_TYPE,/** Package declaration */PACKAGE,/*** Type parameter declaration** @since 1.8*/TYPE_PARAMETER,/*** Use of a type** @since 1.8*/TYPE_USE
}

@Retention

用来表示需要在什么级别保存注解的信息。需要填写枚举类型 RetentionPolicy。

public enum RetentionPolicy {/*** Annotations are to be discarded by the compiler.*/SOURCE,/*** Annotations are to be recorded in the class file by the compiler* but need not be retained by the VM at run time.  This is the default* behavior.*/CLASS,/*** Annotations are to be recorded in the class file by the compiler and* retained by the VM at run time, so they may be read reflectively.** @see java.lang.reflect.AnnotatedElement*/RUNTIME
}

SOURCE 级别则只会在源码阶段存在,编译后就不存在。CLASS 则编译后依然会存在,但是不会jvm中保留。RUNTIME 则会一直存在到代码运行的时候,所以可以通过反射获得

SOURCE 级别简单示例

在常用的lombok中的日志注解:@Slf4j,就是默认在SOURCE级别上,编译之后就会消失,但是会做一定的操作。

@Retention(RetentionPolicy.SOURCE)
@Target({ElementType.TYPE})
public @interface Slf4j {String topic() default "";
}

代码使用@Slf4j:

@Slf4j
public class Recommendation {/*** 需要增强的方法*/public void recommend() {log.info("推荐动漫: 一人之下");}
}

字节码中会将注解变成一句代码

public class Recommendation {//注解生成的语句private static final Logger log = LoggerFactory.getLogger(Recommendation.class);/*** 需要增强的方法*/public void recommend() {log.info("推荐动漫: 小妖怪的夏天");}
}

3. 使用样例

注解定义:

/*** 添加这个注解,可以统计方法的运行时间,* 且携带开发者相关信息* @author cay*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RunningTime {/*** 添加开发负责人的信息,方便对应人员修复bug* @return 开发者相关信息*/String authorMsg();
}

添加注解的方法

/*** 注解使用的样例* @author cay* @since 2023/02/07*/
@Slf4j
public class AnnotationSample {/*** 使用注解的方法*/@RunningTime(authorMsg = "cay")public void recommend() {log.info("推荐电视剧:今日宜加油");}
}

测试单元通过反射获取注解中的信息

/*** 通过反射获取注解的信息*/
@Test
public void testReflect() throws ClassNotFoundException {Class clazz = Class.forName("com.example.demo.simple.AnnotationSample");// 获取该类的所有方法Method[] declaredMethods = clazz.getDeclaredMethods();for (Method declaredMethod : declaredMethods) {if (declaredMethod.isAnnotationPresent(RunningTime.class)) {RunningTime annotation = declaredMethod.getAnnotation(RunningTime.class);String info = annotation.authorMsg();log.info("注解中存储的信息:{}", info);}}
}

注意:

**想要通过反射获取注解,则需要设置为 @Retention(RetentionPolicy.RUNTIME)。那么如果想要处理 SOURCE级别,就可以使用注解处理器Processor **

二、注解和Aop的结合使用

毕竟现在都是使用springboot框架,所以spring也帮我们简化了注解的使用,当自己想要根据注解去做些额外的事情的时候,和aop结合使用会非常的方便。

简单的使用案例,之前写过,就不在赘述了,链接如下

https://blog.csdn.net/weixin_44457062/article/details/128915444

三、AOP

既然提到了Aop,就顺便记录一下Spring中Aop的相关信息。

AOP(Aspect Oriented Programming),即面向切面编程,可以方便的用来统一处理日志、管理事务等,与业务逻辑分离。

相关概念

官网地址:

https://docs.spring.io/spring-framework/docs/current/reference/html/core.html#aop

英文概念:

  • Aspect: A modularization of a concern that cuts across multiple classes. Transaction management is a good example of a crosscutting concern in enterprise Java applications. In Spring AOP, aspects are implemented by using regular classes (the schema-based approach) or regular classes annotated with the @Aspect annotation (the @AspectJ style).
  • Join point: A point during the execution of a program, such as the execution of a method or the handling of an exception. In Spring AOP, a join point always represents a method execution.
  • Advice: Action taken by an aspect at a particular join point. Different types of advice include “around”, “before” and “after” advice. (Advice types are discussed later.) Many AOP frameworks, including Spring, model an advice as an interceptor and maintain a chain of interceptors around the join point.
  • Pointcut: A predicate that matches join points. Advice is associated with a pointcut expression and runs at any join point matched by the pointcut (for example, the execution of a method with a certain name). The concept of join points as matched by pointcut expressions is central to AOP, and Spring uses the AspectJ pointcut expression language by default.
  • Introduction: Declaring additional methods or fields on behalf of a type. Spring AOP lets you introduce new interfaces (and a corresponding implementation) to any advised object. For example, you could use an introduction to make a bean implement an IsModified interface, to simplify caching. (An introduction is known as an inter-type declaration in the AspectJ community.)
  • Target object: An object being advised by one or more aspects. Also referred to as the “advised object”. Since Spring AOP is implemented by using runtime proxies, this object is always a proxied object.
  • AOP proxy: An object created by the AOP framework in order to implement the aspect contracts (advise method executions and so on). In the Spring Framework, an AOP proxy is a JDK dynamic proxy or a CGLIB proxy.
  • Weaving: linking aspects with other application types or objects to create an advised object. This can be done at compile time (using the AspectJ compiler, for example), load time, or at runtime. Spring AOP, like other pure Java AOP frameworks, performs weaving at runtime.

英文的顺序有点乱,自己按照概念排了一下序,大致意思(自己的理解):

  • Join point(连接点): 能都被拦截到的点,在spring中只有方法。在AOP联盟定义的规范里还有可能是字段或构造器
  • Advice(增强或通知): 对 Pointcut 的增强,即我们需要增强的逻辑
  • Pointcut(切入点): 将 Advice 和 Join point 连接起来。
  • Aspect (切面):Advice 、 Pointcut和 Join point 结合
  • Introduction(引介):可以引入额外的方法和字段。
  • Target object :代理的目标对象
  • AOP proxy:动态代理,有 JDK 动态代理和 CGLIB 代理
  • 将切面与其他应用程序类型或对象链接以创建代理的对象。这可以在编译时(例如使用AspectJ编译器)、加载时或运行时完成。与其他纯JavaAOP框架一样,SpringAOP在运行时执行编织。

spring AOP的简单使用

spring已经帮我们封装好了五种 advise,我们可以很简单的实现aop,实现对需要增强方法的增强逻辑。之前对总结过,就不赘述了,之前文章链接:

https://blog.csdn.net/weixin_44457062/article/details/128710179

JDK动态代理

注意:

  • spring默认的是 JDK 动态代理
  • spring boot 2.x 之后,spring boot默认是CGLIB 代理

jdk动态代理是基于接口,生成实现类,复写被增强类里的方法,然后在添加增强方法。

简单使用

  1. 待增强的接口
/*** @author cay*/
public interface IMyAopProxyService {public void myAopProxy();
}
  1. 实现类
/*** @author cay*/
@Slf4j
public class MyAopProxyServiceImpl implements IMyAopProxyService {@Overridepublic void myAopProxy() {log.info("aop proxy......");}
}
  1. 增强类
/*** aop测试的增强类* @author cay*/
@Slf4j
public class MyAopProxyAdvice {/*** 前置增强*/public void beforeMethod() {log.info("before advice ......");}/*** 后置增强*/public void afterMethod() {log.info("after advice ......");}
}
  1. JDK动态代理的测试单元
/*** 测试JDK动态代理*/
@Test
public void testJdkAopProxy() {//需要被增强的类,使用接口接收IMyAopProxyService myAopProxyService = new MyAopProxyServiceImpl();//增强类MyAopProxyAdvice myAopProxyAdvice = new MyAopProxyAdvice();//调用JDK动态代理Object proxy = Proxy.newProxyInstance(myAopProxyService.getClass().getClassLoader(), myAopProxyService.getClass().getInterfaces(),new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {myAopProxyAdvice.beforeMethod();Object invoke = method.invoke(myAopProxyService, args);myAopProxyAdvice.afterMethod();return invoke;}});IMyAopProxyService jdkProxy = (IMyAopProxyService) proxy;log.info("aopProxy: {}", proxy);jdkProxy.myAopProxy();
}

CGLIB代理

CGLIB代理基于类,生成子类,子类覆盖父类方法,并添加增强方法。

spring boot 2.x 之后,spring boot默认是CGLIB 代理,spring的默认代理是 JDK 动态代理。

使用上面JDK代理创建的类,测试单元:

/*** 测试CGLIB代理*/
@Test
public void testCglibAopProxy() {//被增强的类,使用本类接收MyAopProxyServiceImpl myAopProxyService = new MyAopProxyServiceImpl();MyAopProxyAdvice myAopProxyAdvice = new MyAopProxyAdvice();Enhancer enhancer = new Enhancer();enhancer.setSuperclass(myAopProxyService.getClass());enhancer.setCallback(new MethodInterceptor() {@Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {myAopProxyAdvice.beforeMethod();Object invoke = methodProxy.invokeSuper(o, objects);myAopProxyAdvice.afterMethod();return invoke;}});Object o = enhancer.create();MyAopProxyServiceImpl cglibProxy = (MyAopProxyServiceImpl) o;log.info("cglibProxy:{}", cglibProxy);cglibProxy.myAopProxy();
}

相关内容

热门资讯

122.(leaflet篇)l... 听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
Vue使用pdf-lib为文件... 之前也写过两篇预览pdf的,但是没有加水印,这是链接:Vu...
PyQt5数据库开发1 4.1... 文章目录 前言 步骤/方法 1 使用windows身份登录 2 启用混合登录模式 3 允许远程连接服...
Android studio ... 解决 Android studio 出现“The emulator process for AVD ...
Linux基础命令大全(上) ♥️作者:小刘在C站 ♥️个人主页:小刘主页 ♥️每天分享云计算网络运维...
再谈解决“因为文件包含病毒或潜... 前面出了一篇博文专门来解决“因为文件包含病毒或潜在的垃圾软件”的问题,其中第二种方法有...
南京邮电大学通达学院2023c... 题目展示 一.问题描述 实验题目1 定义一个学生类,其中包括如下内容: (1)私有数据成员 ①年龄 ...
PageObject 六大原则 PageObject六大原则: 1.封装服务的方法 2.不要暴露页面的细节 3.通过r...
【Linux网络编程】01:S... Socket多进程 OVERVIEWSocket多进程1.Server2.Client3.bug&...
数据结构刷题(二十五):122... 1.122. 买卖股票的最佳时机 II思路:贪心。把利润分解为每天为单位的维度,然后收...
浏览器事件循环 事件循环 浏览器的进程模型 何为进程? 程序运行需要有它自己专属的内存空间࿰...
8个免费图片/照片压缩工具帮您... 继续查看一些最好的图像压缩工具,以提升用户体验和存储空间以及网站使用支持。 无数图像压...
计算机二级Python备考(2... 目录  一、选择题 1.在Python语言中: 2.知识点 二、基本操作题 1. j...
端电压 相电压 线电压 记得刚接触矢量控制的时候,拿到板子,就赶紧去测各种波形,结...
如何使用Python检测和识别... 车牌检测与识别技术用途广泛,可以用于道路系统、无票停车场、车辆门禁等。这项技术结合了计...
带环链表详解 目录 一、什么是环形链表 二、判断是否为环形链表 2.1 具体题目 2.2 具体思路 2.3 思路的...
【C语言进阶:刨根究底字符串函... 本节重点内容: 深入理解strcpy函数的使用学会strcpy函数的模拟实现⚡strc...
Django web开发(一)... 文章目录前端开发1.快速开发网站2.标签2.1 编码2.2 title2.3 标题2.4 div和s...