张麻子:汤师爷,翻译翻译,什么叫AOP?

汤师爷:这还用翻译。

张麻子:我让你翻译给我听,什么叫AOP?

汤师爷:不用翻译,切面编程就是AOP啊。

黄四郎:难道你听不懂什么叫AOP?

张麻子:我就想让你翻译翻译,什么叫AOP!

汤师爷:AOP嘛。

张麻子:翻译出来给我听,什么他妈的叫AOP!什么他妈的叫他妈的AOP!

汤师爷:什么他妈的叫AOP啊?

黄四郎:AOP就是Aspect Oriented Programming,面向切面编程!明白了吗?

汤师爷:这就是AOP啊。

张麻子:翻译翻译。

汤师爷:...

汤师爷:AOP就是Aspect Oriented Programming!面向切面编程!面向!切面!横着切!切面!

张麻子:哈,大哥这是他妈的AOP啊,小弟我马上给个三连。

下面我们好好翻译一下AOP切面编程。

老规矩,在学习切面面编程之前要有前置知识:

终于搞懂动态代理了!

目标

• 动态代理搞明白,AOP就明白了

• 学会在开发中使用Spring的AOP技术

概念

AOP 是 Aspect Oriented Programming,和OOP(Object Oriented Programming)一词之差,OOP强调万物皆是对象,那AOP呢?

要真正理解 AOP 就要理解 AOP 的核心:Aspect

WTF is Aspect?

我们把aspect翻译成切面,但是切面这个词对应中文语义其实很难理解到位。

我们换种解释,aspect我们理解为事物的某个方面、某个视角

与面向对象思想相对,对象强调一个整体,一个人站在你面前,我们称之为对象。

而aspect强调功能化、模块化、关注点分离

天气变冷了,人要多穿衣服,上帝控制这么多人的对象,总不能一件一件穿吧?

所以AOP思想就是把天气变冷穿衣服这件事抽离出来,模块化,单独进行关注,然后经过编码实现后,上帝就可以进行一键穿衣,节省了大量工作。

放在实际开发中,我们以最常见的日志打印为例。

我们系统中有200个Controller,都要打印请求日志,那没有AOP思想的实践的话,我们只能一个一个在Controller里编写一遍又一遍的重复代码,有了AOP思想,就可以考虑把日志打印这件事抽离出来做成单独的业务,实现一劳永逸。

那怎么实现呢?

那就要靠我们的动态代理了。

什么?你还没看动态代理?

上帝在吗?把这个人的棉衣扒了。

Spring实现AOP

我们知道框架的存在意义是用来简化开发的。

这里Spring简化了什么呢?

自然是我们在动态代理部分编写的一大堆要么看不懂,看懂了又不想写的代码。

AOP作为Spring的左膀右臂之一,自然对这部分加以简化。

但Spring那一大堆xml也是够够的。

所以SpringBoot才是我们永远的家。

废话不多说直接上代码示例,在AOP代码的编写中提出问题,解决问题,那么最后就算是学会了。

引入依赖

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--aop-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
    <scope>test</scope>
</dependency>

编写代码

编写这部分代码的逻辑也是非常符合我们的认知逻辑的。

我们前边已经说清楚AOP是做什么的了。

那我们编写代码就要做3件事:

  • • 抽离出功能模块——定义切面

  • • 确认功能代码加在哪——定义切点

  • • 确认功能代码什么时候执行——选择通知类型

我们先把controller给出来:

@RestController
public class AopController {     @RequestMapping("/hello")
    public String sayHello(){
        System.out.println("hello");
        return "hello";
    }
}

定义一个切面

也就是抽离出功能模块

先随便写个类。

然后就直接一个@Aspect就行了,那这个类就是一个切面类。

还要再加一个@Component将该类纳入Ioc容器。

就这么简单,狗来了都会写。

@Aspect
@Component
public class AopAdvice {
}

定义一个切点

也就是确认功能代码加在哪

先随便写一个方法。

然后就直接一个@Pointcut就行了,那这个方法就是一个切点。

还要再加上表达式,让系统知道代码加到什么位置。

@Aspect
@Component
public class AopAdvice { @Pointcut("execution (* com.example.aop.controller.*.*(..))")
public void test() {
}
}

这时候有同学问:

啊这个execution是什么?

里面那又是一坨什么?

根本看不懂。

举报了。

这个我只能说,这是固定的表达式,是规定。

规定什么?

看规定之前先记住:表达式一定从右往左匹配。

看规定之前先记住:表达式一定从右往左匹配。

看规定之前先记住:表达式一定从右往左匹配。

execution(访问修饰符(可省略) 方法返回值 包名.类名.方法名(参数))
参数:..代表任何参数
方法: *代表任何方法
类名: *代表所有类
包名: *代表所有包 ..代表子孙包
返回值: *代表所有类型返回值

具体的写法实际五花八门,而且除了execution还有一大堆,为了不让大脑过度疲劳,我们一次只有一个目标:

会用,但不精通。

选择通知类型

也就是确认功能代码什么时候执行

下面就是通知类型5种,前3种比较常用:

前置通知(@Before):在目标方法调用之前调用通知

后置通知(@After):在目标方法完成之后调用通知

环绕通知(@Around):在被通知的方法调用之前和调用之后执行自定义的方法

返回通知(@AfterReturning):在目标方法成功执行之后调用通知

异常通知(@AfterThrowing):在目标方法抛出异常之后调用通知

代码如下:

@Aspect
@Component
public class AopAdvice { @Pointcut("execution (* com.example.aop.controller.*.*(..))")
public void test() { } @Before("test()")
public void beforeAdvice() {
System.out.println("beforeAdvice...");
} @After("test()")
public void afterAdvice() {
System.out.println("afterAdvice...");
} @Around("test()")
public void aroundAdvice(ProceedingJoinPoint joinPoint) {
System.out.println("before");
try {
joinPoint.proceed();
} catch (Throwable t) {
t.printStackTrace();
}
System.out.println("after");
}
}

从以上代码中可以看出,@Before和@After都已经简单到不能再简单了。

我们需要说一下这个@Around 环绕通知

joinPoint.proceed()这行代码我们就理解为我们需要增强的那个方法的替身就行了。

这也不是想说的,这里我们主要讲一下 ProceedingJoinPoint joinPoint 这个参数。

ProceedingJoinPoint 是一个接口,也就是说这里实际是使用了多态,这不重要。

ProceedingJoinPoint 继承了 JoinPoint 这个接口。

这个JoinPoint有2个方法是我们需要说的。

Object getTarget();
Signature getSignature();

getTarget() 方法返回的是目标对象,即那个被增强方法所属的对象实例。

getSignature() 方法返回的是连接点的签名,即关于被调用的方法(或访问的字段等)的静态信息,如方法名、返回类型、参数类型等。

那么通过这2个方法,就能获取到所有的对象信息和方法信息,那么能做的事就太多了。

但是我们这里不展示更复杂的案例。

依然坚持我们本篇的策略:

懂点,但不多。

会用,但不精。

测试

启动项目,浏览器访问:

http://localhost:8080/hello

运行结果:

before
beforeAdvice...
hello
afterAdvice...
after

往期推荐:

● 0.o?让我看看怎么个事儿之SpringBoot自动配置

终于搞懂动态代理了!

● 学会@ConfigurationProperties月薪过三千

● 学一点关于JVM类加载的知识

● Java注解,看完就会用

● Java反射,看完就会用

师爷,翻译翻译什么叫AOP的更多相关文章

  1. (翻译) 使用Unity进行AOP对象拦截

    Unity 是一款知名的依赖注入容器( dependency injection container) ,其支持通过自定义扩展来扩充功能. 在Unity软件包内 默认包含了一个对象拦截(Interce ...

  2. R语言包翻译——翻译

    Shiny-cheatsheet                                                                                     ...

  3. Python函数:使用谷歌翻译翻译英语字符串

    代码是同事写的,我把它单独抠出来,可以作为工具函数使用.当然,性能还是个问题,有待解决. import random import cookielib import urllib import url ...

  4. C# 调用百度翻译Api

    这是简单的界面.用的是wpf,winform也可以 具体的操作类 public partial class MainWindow : Window { string url = "" ...

  5. 四大高质量且实用的chrome翻译插件推荐

    Google英译汉的质量怎么样?日常生活用语翻译还可以,但是一到专业性术语就歇菜了,翻译出来的东西简直就是惨不忍睹,惨绝人寰..对于酷爱英语学习又有强迫症的患者来说,一款既实用又方便,无疑就是雪中送炭 ...

  6. 干货|人人都是翻译项目的Master

    在平时的工作中,我们都会经常查阅一些英文文档来解决平时遇到的问题和拓宽视野.看到好的文章或者书籍有没有想要和小伙伴分享的冲动,那么我们一起来翻译吧- 翻译主张 "信 达 雅" .& ...

  7. IDEA Translation插件中有道智云(有道翻译)应用ID,密钥申请教程

    登录链接 该登录登录,该注册注册(信息随意填写) 自然语言翻译=>翻译实例=>创建实例(信息随意填写) QQ截图20170701231552.png 应用管理=>我的应用=>创 ...

  8. 完善chrome翻译插件ChaZD,支持有道智云api

    首先放上该项目的github地址:https://github.com/codethereforam/ChaZD 之前想找一个chrome支持划词翻译的插件,最终在知乎上看到了这个回答,推荐的是Cha ...

  9. selenium批量翻译

    Python爬虫视频教程零基础小白到scrapy爬虫高手-轻松入门 https://item.taobao.com/item.htm?spm=a1z38n.10677092.0.0.482434a6E ...

  10. 论文翻译——Rapid 2D-to-3D conversion——快速2D到3D转换

    https://blog.csdn.net/qq_33445835/article/details/80143598  目前想做一个关于2D转3D的项目,由于国内资料比较少而且大部分都是基于国外的研究 ...

随机推荐

  1. Vue源码学习(十七):实现computed计算属性

    好家伙,本章我们尝试实现computed属性 0.完整代码已开源 https://github.com/Fattiger4399/analytic-vue.git 1.分析 1.1computed的常 ...

  2. 详细一些的vue生命周期

    如果你和我一样,以前对vue生命周期的理解仅限于生命周期钩子,那么本文可以让我们更深入一层,理解vue在生命周期各个阶段所做的事,对我们对vue的理解和使用很有好处. (1)通过new Vue()创建 ...

  3. 经典卷积神经网络LeNet&AlexNet&VGG

    LeNet LeNet-5是一种经典的卷积神经网络结构,于1998年投入实际使用中.该网络最早应用于手写体字符识别应用中.普遍认为,卷积神经网络的出现开始于LeCun等提出的LeNet网络,可以说Le ...

  4. 年度盘点,四年的精华合集「GitHub 热点速览」

    今年是 GPT 年,无论是 GitHub 还是朋友圈还是技术平台,即便你不关心 GPT 的发展情况,同大模型.AI 相关的项目总能进入你的信息流.到这期为止,热度速览也连载了四年,从一开始习惯看 Gi ...

  5. SpringBoot对象拷贝

    目录 概述 定义实体类 Car size carInfo 造测试数据 Spring BeanUtils Apache BeanUtils Cglib BeanCopier MapStruct 性能测试 ...

  6. ajax与thymeleaf分别实现数据传输

    小杰笔记篇: 1:第一种:利用Model和thymeleaf引擎来完成: Controller层: @CrossOrigin//解决跨域问题 @Controller public class User ...

  7. shiro基于角色URL进行鉴权

    前言 shiro基于URL进行鉴权,网上有很多,但是多数都是copy不排版,眼睛都看花了,还不如自己看看源码. 2021年1月14日21:23:49最新的shiro是1.7,使用时发现了首次访问的一个 ...

  8. 通用 Mapper 的批量插入实现

    具体的 SQL 模板实现如下所示: import org.apache.ibatis.mapping.MappedStatement; import tk.mybatis.mapper.MapperE ...

  9. QT OpenGLWidget高分屏适配时出现的问题

    参考官方文档,为了解决4K屏幕显示QT界面时出现窗口变小分辨率不匹配的问题,我们可以在 QApplication a(argc, argv); 这句之前添加: #if (QT_VERSION > ...

  10. 视频编码耗时长、编码帧发送失败…DVPP视频编码问题典型案例分析

    摘要:本期就分享几个关于DVPP视频编码问题的典型案例,并给出原因分析及解决方法 本文分享自华为云社区<DVPP媒体数据处理视频编码问题案例>,作者:昇腾CANN. DVPP(Digita ...