Google Guice 5:AOP

news2024/9/24 7:01:52

1. AOP

1.1 实际开发中面临的问题

  • 在实际开发中,经常需要打印一个方法的执行时间,以确定是否存在慢操作

  • 最简单的方法,直接修改已有的方法,在finnally语句中打印耗时

    @Override
    public Optional<Table> getTable(String databaseName, String tableName)
    {
        long startTime = System.currentTimeMillis();
    
        try {
            ... // 省略关键代码
        	return result;
        }
        ... // 省略catch语句
        finally {
            long consumedTime = System.currentTimeMillis() - startTime;
            logger.info("getTable() -- databaseName: %s, tableName: %s, consumed time: %dms", databaseName, tableName, consumedTime);
        }
    }
    
  • 随着打印耗时的需求增多,你会发现整个应用程序中存在很多打印耗时的代码

  • 这些代码如果按照OOP(Object Oriented Programming)编程的思想,是无法抽象出来复用的,因为它与方法的执行是紧耦合的

  • 而且随着需求的增多,安全检查、权限校验、审计日志等附加功能的代码,会使得整个方法越来越臃肿,以致难以阅读

  • 同时,如果附加功能的逻辑发生变化,将牵一发而动全身

1.2 AOP编程

  • 这时,我们希望有一种编程范例,能将这些公共的代码从方法中剥离出来,并切入到有需要的方法中

  • 这种范例叫做AOP(Aspect-Oriented Programming ,面向切面编程)

  • 它将问题划分成了不同的面(Aspect),在程序编译或运行阶段可以将这些面插入到需要的地方,以实现辅助功能与核心逻辑的解耦以及代码的复用

  • 维基百科对AOP的描述如下:

    1. In computing, aspect-oriented programming (AOP) is a programming paradigm that aims to increase modularity by allowing the separation of cross-cutting concerns.
      在计算机领域,AOP是一种允许横切关注以增加代码模块化的编程范例
    2. It does so by adding behavior to existing code (an advice) without modifying the code itself, instead separately specifying which code is modified via a “pointcut” specification, such as “log all function calls when the function’s name begins with ‘set’”.
      它可以在不修改代码本身的情况下,向已有的代码添加行为(an advice),通过 "pointcut"规范来单独指定修改哪些代码,例如:记录所有以set开头的函数的调用
    3. This allows behaviors that are not central to the business logic (such as logging) to be added to a program without cluttering the code core to the functionality.
      这样可以允许将不属于业务核心逻辑的行为(如日志打印)添加到程序中,并不会导致功能的代码核心紊乱
  • 上述的第二点,笔者觉得维基百科的描述可能存在一定问题:

    • advice,通知,是对被拦截到的方法需要执行的代码(又叫增强),可以是前置、后置、最终、异常、环绕通知五类。以环绕通知为例,就是在方法执行前后需要执行的代码
    • pointcut: joint point是连接点的意思,即可能需要注入切面的地方;而pointcut用于定义需要特殊处理的连接点

1.3 Srping AOP的常见术语

  • 使用过Spring进行AOP编程的小伙伴,可能会注意到Aspect、Pointcut、Advice(Before、After、AfterReturning、AfterThrowing、Around)这些类似的关键字

  • 在Spring AOP中,有很多术语,但最常使用的还是上面三种术语

  • 下面是一段打印controller请求参数的AOP代码

    @Component
    @Aspect // 表示这是一个切面
    public class ControllerParamsLogAspect {
        private static final Logger logger = LoggerFactory.getLogger(ControllerParamsLogAspect.class);
        
    	// 定义切点,匹配com.sunrise.controller包下所有的public方法,方法的参数任意
        @Pointcut("execution(public * com.sunrise.controller..*.*(..))")
        public void logAspect() {
        }
        
        // 前置通知,在执行controller方法前,打印方法入参
        @Before("logAspect()")
        public void doBefore(JoinPoint point) {
            // 构建打印内容
            JSONObject content = new JSONObject();
            try {
            	... // 省略核心代码
            	logger.error("API请求信息:{}", CommonUtils.jsonObjectToString(content));
            } catch (Exception exception) {
                ... // 省略异常处理代码
            }
        } 
    }
    
  • 以上与AOP有关的注解,需要的maven依赖如下:

    <dependency>
    	<groupId>org.aspectj</groupId>
    	<artifactId>aspectjrt</artifactId>
    	<version>1.8.14</version>
    </dependency>
    
  • 其余术语概念可以参考:

    • Spring AOP详解
    • Spring 学习笔记(四):Spring AOP
  • PS: 阅读理论知识如果让你觉得迷糊,可以尝试直接看一段代码或者写一个小demo

2. AOP的实现方法

2.1 笔者的理解

  • 通过对AOP基础知识的学习,在笔者心中AOP的实现方法,就应该像动态代理一样:以某种方式调用方法(如反射),并在方法的执行前后添加一些代码,从而可以在不修改已有业务逻辑的情况下增强方法

  • 这是使用JDK动态代理实现的,打印方法耗时的核心代码:

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        long startTime = System.currentTimeMillis();
        try {
            return method.invoke(animal, args);
        } finally {
            long diff = System.currentTimeMillis() - startTime;
            System.out.printf("%s() called, consumed %dms\n", method.getName(), diff);
        }
    }
    
  • 然而,动态代理是有针对性的,AOP应该是有普适性的

    • cglib动态代理虽然能代理一个类,但是后续对该类的访问,都需要转为对proxy的访问
    • AOP则可以在满足特定规则的所有方法上使用,例如,只要对方法添加指定注解,Aspect则可以应用到这些方法上;又或者,所以以set开头的方法,都可以打印审计日志以记录数据的变化
  • 让笔者不认同的是,有些博客宣称可以使用动态代理实现AOP,这明显是忽略了AOP的普适性

  • 当然,这也可能是笔者对动态代理的理解不够深入 🤣 🤣 🤣

  • Guice使用拦截器机制实现AOP,笔者也比较赞同这种思想

2.2. Guice对AOP的支持

  • 使用AOP实现一个打印方法耗时的功能,在目标方法执行结束后,系统会自动打印该方法的耗时

  • 定义一个注解,使用该注解的方法,都将获得耗时打印的附加能力

    @Target({METHOD})
    @Retention(RUNTIME)
    public @interface PrintExecutionTime {
    }
    
  • 定义一个打印方法耗时的拦截器,它实现了org.aopalliance.intercept.MethodInterceptor接口

    public class ExecutionTimePrinter implements MethodInterceptor {
        @Override
        public Object invoke(MethodInvocation invocation) throws Throwable {
            long startTime = System.currentTimeMillis();
            try {
                return invocation.proceed();
            } finally {
                long consumedTime = System.currentTimeMillis() - startTime;
                System.out.printf("call %s() and consumed %dms\n", invocation.getMethod().getName(), consumedTime);
            }
        }
    }	
    
  • 将拦截器应用到计算器的某些方法上

    public class Calculator {
        private static final Random random = new Random();
    
        @PrintExecutionTime
        public int add(int a, int b) throws Throwable {
            // 随机休眠一段时间
            Thread.sleep(random.nextInt(2000));
            return a + b;
        }
    
        public int sub(int a, int b) {
            return a - b;
        }
    
        @PrintExecutionTime
        public int div(int a, int b) throws Throwable {
            // 随机休眠一段时间
            Thread.sleep(random.nextInt(2000));
            return a / b;
        }
    
        public long mul(int a, int b) {
            return a * b;
        }
    }
    
  • 在Module中定义pointcut,即拦截器将作用到满足什么规则的方法上

    public class AopModule extends AbstractModule {
        @Override
        protected void configure() {
        	// ExecutionTimePrinter将作用到任意类的、使用了@PrintExecutionTime的方法上
            bindInterceptor(Matchers.any(), Matchers.annotatedWith(PrintExecutionTime.class),
                    new ExecutionTimePrinter());
        }
    }
    
  • 从Guice中获取一个计算器实例,执行其方法以验证拦截器是否生效

    public static void main(String[] args) throws Throwable {
        Injector injector = Guice.createInjector(new AopModule());
        Calculator calculator = injector.getInstance(Calculator.class);
        int a = 12, b = 24;
        System.out.printf("%d + %d = %d\n", a, b, calculator.add(a, b));
        System.out.printf("%d - %d = %d\n", a, b, calculator.sub(a, b));
        System.out.printf("%d / %d = %d\n", b, a, calculator.div(b, a));
        System.out.printf("%d * %d = %d\n", a, b, calculator.mul(a, b));
    }
    
  • 最终,添加了@PrintExecutionTime的add()div()方法执行结束后将打印耗时

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

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

相关文章

中级嵌入式系统设计师2014下半年下午试题与答案解析

中级嵌入式系统设计师2014下半年下午试题与答案解析 试题一 阅读下列说明和图,回答下列问题。 [说明] ATM自动取款机系统是一个由终端机、ATM系统、数据库组成的应用系统,具有提取现金、查询账户余额、修改密码及转账等功能。ATM自动取款机系统用例图如图1所示。

win11开始菜单增强工具:StartAllBack

StartAllBack是一款Windows11开始菜单增强工具&#xff0c;在任务栏上为Windows 11恢复经典样式的Windows 7主题风格开始菜单&#xff0c;主要功能包括&#xff1a;恢复和改进开始菜单样式、个性化任务栏、资源管理器等功能。软件功能恢复和改进任务栏在任务图标上显示标签调整…

【Spring】通过JdbcTemplate实现CRUD操作

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 通过JdbcTemplate实现 增删查改一、添加相关依…

python-下载某短视频平台视频(高清无水印)

python-下载某短视频平台音视频&#xff08;高清无水印&#xff09;前言1、获取视频 url2、发送请求3、数据解析4、本地保存5、完整代码前言 1、Cookie中文名称为小型文本文件&#xff0c;指某些网站为了辨别用户身份而储存在用户本地终端&#xff08;Client Side&#xff09;…

RTMP的工作原理及优缺点

一.什么是RTMP&#xff1f;RTMP&#xff08;Real-Time Messaging Protocol&#xff0c;实时消息传输协议&#xff09;是一种用于低延迟、实时音视频和数据传输的双向互联网通信协议&#xff0c;由Macromedia&#xff08;后被Adobe收购&#xff09;开发。RTMP的工作原理是&#…

Windows 11 网卡MAC地址 | 机器地址 | 网络地址 为 0 | 00-00-00-00-00-00?手动修复……

一位同事反映&#xff0c;他的电脑今天上班开机无法上网&#xff0c;上周末还正常&#xff0c;请我帮忙检修。该同事的电脑安装的是Windows 11&#xff0c;检查网络连接的详细信息&#xff0c;发现IP地址、网关、DNS参数都正常&#xff0c;但物理地址为00-00-00-00-00-00。另外…

力扣每日一题(2023年2月)

2023年2月期每日一题第一天 &#xff08;2325. 解密消息&#xff09;第十六天&#xff08;2341. 数组能形成多少数对&#xff09;第十七天 &#xff08;1139. 最大的以 1 为边界的正方形&#xff09;第十八天 &#xff08;1237. 找出给定方程的正整数解&#xff09;第十九天 &a…

再度盈利,搜狐稳了?

2016年在宣布要用3年时间回归互联网舞台中心之后&#xff0c;很长一段时间内张朝阳积极活跃在各种社交媒体上&#xff0c;完全是一派“积极出山”的姿态。而后畅游从美股退市&#xff0c;搜狗“卖身”腾讯&#xff0c;一系列的收缩动作又似乎是在逐渐远离喧嚣。而在最近三年&am…

Zilliz @ GAIDC |Milvus:生产级 AI 数据库探索

2月26日&#xff0c;全球人工智能开发者先锋大会— AI 数据与开源论坛在上海举行&#xff0c;Zilliz 主任工程师刘力在现场发表了名为《Milvus&#xff1a;生产级 AI 数据库探索》的演讲。刘力开篇便提到&#xff0c;Milvus 致力于打造更为完善的 AI 生态体系。随后&#xff0c…

阿赵的MaxScript学习笔记分享五《UI组件使用篇》

大家好&#xff0c;我是阿赵。这个专题的内容比较多&#xff0c;所以抓紧时间更新。这是第五篇&#xff0c;UI组件使用篇。 这里主要是把maxscript的rollout窗体支持的UI组件列举一下&#xff0c;并每个组件写一个使用范例&#xff0c;以方便理解和查看。 1、位图bitmap 可以指…

Java中常用的七种队列你了解多少?

文章目录Java中常用的七种队列你了解多少?ArrayBlockingQueue队列如何使用&#xff1f;添加元素到队列获取队列中的元素遍历队列LinkedBlockingQueue队列如何使用&#xff1f;1. 创建SynchronousQueue对象2. 添加元素到队列3. 获取队列中的元素4. 遍历队列SynchronousQueue队列…

Unity Lighting -- Unity的光源简介

在主菜单栏中&#xff0c;点击Window -> Rendering -> Light Explorer打开光源管理器&#xff0c;这个标签页可以看到场景中所有的光源&#xff0c;包括每个光源的类型&#xff0c;形状&#xff0c;模式&#xff0c;颜色&#xff0c;强度&#xff0c;阴影等信息。 在主菜…

android unit test mock框架使用记录

写在前面 之前上班时&#xff0c;开发一个功能之后&#xff0c;还需要编写测试用例&#xff0c;使用的框架是mock。 为什么防止以后用到时忘了&#xff0c;在这里记录一下。 由于团队没有人使用Espresso进行unit test&#xff0c;所以本人对该框架并不熟悉。想了解该框架的使用…

WPF布局控件之DockPanel

DockPanel DockPanel&#xff0c;英文释义为停靠面板&#xff0c;那是怎么个停靠法呢&#xff1f;如下&#xff1a; <Window x:Class"LearnLayout.DockPanelWin"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http:/…

数字图像处理

文章目录图像复原上升阶跃边缘、下降阶跃边缘、脉冲状边缘和屋顶状边缘曲线及其一阶导数和二阶导数有哪些特征&#xff1f;Hough变换的基本思想是什么&#xff1f;基本概念图像增强灰度变换直方图&#xff1a;直方图特点matlab代码空间域滤波平滑空间滤波均值滤波器&#xff1a…

python如何实现多线程

今天本来打算学习学习多进程的&#xff0c;但是由于我现在的电脑没有Linux系统&#xff0c;无法通过Linux系统编辑一些多进程的程序&#xff0c;因此我打算从多线程入手。 多线程 我们的程序一般都是多任务的&#xff0c;如果你没有好好的利用好&#xff0c;运行时就会出现卡…

【读论文】TCL: an ANN-to-SNN Conversion with Trainable Clipping Layers

DAC 2021 背景 通过ANN2SNN的方法得到的SNN中&#xff0c;存在准确性和延迟之间的一种权衡关系&#xff0c;在较大的数据集&#xff08;如ImageNet&#xff09;上可能会有较高的延迟。 主要贡献 分析了转换后SNN精度与延迟之间存在权衡关系的原因&#xff0c;并指出了如何缓…

6587: 计算分段函数

描述本题目要求计算下列分段函数f(x)的值&#xff1a;输入输入在一行中给出实数x。输出在一行中按“f(x) result”的格式输出&#xff0c;其中x与result都保留两位小数。样例输入10样例输出f(10.00) 3.16提示C/C可在头文件中包含math.h&#xff0c;并调用sqrt函数求平方根&am…

JAVA入门教程||Java Scanner 类||Java 异常处理

Java Scanner 类 java.util.Scanner是Java5的新特征&#xff0c;我们可以通过 Scanner 类来获取用户的输入。 下面是创建 Scanner 对象的基本语法&#xff1a; Scanner s new Scanner(System.in); 接下来我们演示一个最简单的的数据输入&#xff0c;并通过 Scanner 类的 nex…

产品分析丨豆瓣APP

本文将从以下几个方面进行分析&#xff1a;1. 产品功能结构2. 竞品分析3. 用户分析4. 用户调研5. 功能分析与优化方案6. 总结01 产品功能结构产品架构由豆瓣的产品架构图可看出&#xff0c;豆瓣是兼具书影音的评分系统和兴趣社区&#xff0c;以广告、知识付费和电商业务作为商业…