第十六章 Spring之假如让你来写AOP——Pointcut(切点)篇

Spring源码阅读目录

第一部分——IOC篇

第一章 Spring之最熟悉的陌生人——IOC
第二章 Spring之假如让你来写IOC容器——加载资源篇
第三章 Spring之假如让你来写IOC容器——解析配置文件篇
第四章 Spring之假如让你来写IOC容器——XML配置文件篇
第五章 Spring之假如让你来写IOC容器——BeanFactory和FactoryBean
第六章 Spring之假如让你来写IOC容器——Scope和属性填充
第七章 Spring之假如让你来写IOC容器——属性填充特别篇:SpEL表达式
第八章 Spring之假如让你来写IOC容器——拓展篇
第九章 Spring之源码阅读——环境搭建篇
第十章 Spring之源码阅读——IOC篇

第二部分——AOP篇

第十一章 Spring之不太熟的熟人——AOP
第十二章 Spring之不得不了解的内容——概念篇
第十三章 Spring之假如让你来写AOP——AOP联盟篇
第十四章 Spring之假如让你来写AOP——雏形篇
第十五章 Spring之假如让你来写AOP——Joinpoint(连接点)篇
第十六章 Spring之假如让你来写AOP——Pointcut(切点)篇
第十七章 Spring之假如让你来写AOP——Advice(通知)上篇
第十八章 Spring之假如让你来写AOP——Advice(通知)下篇
第十九章 Spring之假如让你来写AOP——番外篇:Spring早期设计
第二十章 Spring之假如让你来写AOP——Aspect(切面)篇
第二十一章 Spring之假如让你来写AOP——Weaver(织入器)篇
第二十二章 Spring之假如让你来写AOP——Target Object(目标对象)篇
第二十三章 Spring之假如让你来写AOP——融入IOC容器篇
第二十四章 Spring之源码阅读——AOP篇

第三部分——事务篇

第二十五章 Spring之曾经的老朋友——事务
第二十六章 Spring之假如让你来写事务——初稿篇
第二十七章 Spring之假如让你来写事务——铁三角篇
第二十八章 Spring之假如让你来写事务——属性篇
第二十九章 Spring之假如让你来写事务——状态篇
第三十章 Spring之假如让你来写事务——管理篇
第三十一章 Spring之假如让你来写事务——融入IOC容器篇
第三十二章 Spring之源码阅读——事务篇

第四部分——MVC篇

第三十三章 Spring之梦开始的地方——MVC
第三十四章 Spring之假如让你来写MVC——草图篇
第三十五章 Spring之假如让你来写MVC——映射器篇
第三十六章 Spring之假如让你来写MVC——拦截器篇
第三十七章 Spring之假如让你来写MVC——控制器篇
第三十八章 Spring之假如让你来写MVC——适配器篇
第三十九章 Spring之假如让你来写MVC——番外篇:类型转换
第四十章 Spring之假如让你来写MVC——ModelAndView篇
第四十一章 Spring之假如让你来写MVC——番外篇:数据绑定
第四十二章 Spring之假如让你来写MVC——视图篇
第四十三章 Spring之假如让你来写MVC——上传文件篇
第四十四章 Spring之假如让你来写MVC——异常处理器篇
第四十五章 Spring之假如让你来写MVC——国际化篇
第四十六章 Spring之假如让你来写MVC——主题解析器篇
第四十七章 Spring之假如让你来写MVC——闪存管理器篇
第四十八章 Spring之假如让你来写MVC——请求映射视图篇
第四十九章 Spring之假如让你来写MVC——番外篇:属性操作
第五十章 Spring之假如让你来写MVC——融入IOC容器篇
第五十一章 Spring之源码阅读——MVC篇

第五部分——Boot篇

第五十二章 Spring之再进一步——Boot
第五十三章 Spring之假如让你来写Boot——环境篇
第五十四章 Spring之假如让你来写Boot——注解篇(上)
第五十五章 Spring之假如让你来写Boot——注解篇(下)
第五十六章 Spring之假如让你来写Boot——SPI篇
第五十七章 Spring之假如让你来写Boot——配置文件篇(上)
第五十八章 Spring之假如让你来写Boot——配置文件篇(下)
第五十九章 Spring之假如让你来写Boot——番外篇:再谈Bean定义
第六十章 Spring之假如让你来写Boot——自动装配篇
第六十一章 Spring之假如让你来写Boot——番外篇:杂谈Starter
第六十二章 Spring之假如让你来写Boot——番外篇:重构BeanFactory
第六十三章 Spring之假如让你来写Boot——番外篇:再谈ApplicationContext
第六十四章 Spring之假如让你来写Boot——内嵌Web容器篇
第六十五章 Spring之假如让你来写Boot——Main方法启动篇
第六十六章 Spring之最终章——结语篇



前言

    对于Spring一直都是既熟悉又陌生,说对它熟悉吧,平时用用没啥问题,但面试的时候被问的一脸懵逼,就很尴尬,都不好意思在简历上写着熟悉Spring了
在这里插入图片描述

    所以决定花点时间研究研究Spring的源码。主要参考的书籍是:《Spring源码深度解析(第2版)》、《Spring揭秘》、《Spring技术内幕:深入解析Spring架构与设计原理(第2版)》


    书接上回,在上篇 第十五章 Spring之假如让你来写AOP——Joinpoint(连接点)篇 中,A君Joinpoint(连接点) 部分搞定了。接下来看看 A君 会有什么骚操作吧

尝试动手写IOC容器

    出场人物:A君(苦逼的开发)、老大(项目经理)

    背景:老大要求A君在一周内开发个简单的 IOC容器

    前情提要: A君Joinpoint(连接点) 部分搞定了 。。。

第十八版 Pointcut(切点)

    一大早,A君 就被拉到会议室。。。

    “ A君,昨天的 Joinpoint(连接点) 感觉如何?是不是觉得太简单了。” 老大 问道

    “额,还是有些难度的。” A君 回道,开玩笑,这种事怎么可能觉得简单,一旦承认了,后边的活只会越来越难

    “嘿,今天可比昨天难些哦” 老大 笑眯眯的说到,他可不信 A君 所谓的 ‘难’,顿一下,继续说道 “我看你写的 Pointcut(切点) 部分,显然偷懒了,连接口都没设计。”

    “ Pointcut(切点) 接口要怎么设计呢?难道直接定义个match方法吗?” A君 疑惑道

    “这个问题问得好。” 老大 赞道,“确实,直接定义个match方法也是可以的,但是有些情况下,用户只想根据类名匹配,或者只想根据方法名匹配,我们不妨把类和方法匹配分离,这样可以提供更自由组合匹配方式。其次,我们需要稍微考虑一下效率问题,如果这个系统很庞大,涉及到成千上万个类,这时候一个个类匹配过去,效率会很慢的。可以考虑先按类过滤,类匹配了,在匹配具体方法。”

    “明白!” 听完 老大 的分析思路, A君 的眼神逐渐清澈起来了。由衷感叹道:“这也许就是设计之美吧!”

    实现方案 老大 已经拍板了,A君 只需要照做就行了。既然分为方法匹配和类匹配,需要分别定义接口,A君 定义 ClassFilter 作为类匹配的顶级接口,代码如下:

@FunctionalInterface
public interface ClassFilter {
   
    ClassFilter TRUE = TrueClassFilter.INSTANCE;

    boolean matches(Class<?> clazz);
}

接下来需要定义具体实现,是否是某个类的子类,是类匹配最常见的作法,所以 A君 提供 RootClassFilter 类,作为其实现。代码如下:

import com.hqd.ch03.v18.aop.ClassFilter;
import lombok.AllArgsConstructor;

/**
 * 子类匹配器
 */
@AllArgsConstructor
public class RootClassFilter implements ClassFilter {
   
    private final Class<?> clazz;


    @Override
    public boolean matches(Class<?> clazz) {
   
        return this.clazz.isAssignableFrom(clazz);
    }
}

单一个子类匹配显然满足不了复杂的业务环境,所幸,A君Aspectj 整合进来了,可以借用切点表达式匹配,来应对更为复杂的需求。不过,切点表达式不支持 and、or、not ,这里做个简单的优化,替换成对应的 &&、||、! 即可。A君 定义 TypePatternClassFilter,作为切点表达式的实现,代码如下:

import com.hqd.ch03.v18.aop.ClassFilter;
import org.apache.commons.lang3.StringUtils;
import org.aspectj.weaver.tools.PointcutParser;
import org.aspectj.weaver.tools.TypePatternMatcher;
import org.junit.Assert;

public class TypePatternClassFilter implements ClassFilter {
   
    private String typePattern = "";
    private TypePatternMatcher aspectJTypePatternMatcher;

    public TypePatternClassFilter() {
   
    }

    public TypePatternClassFilter(String typePattern) {
   
        setTypePattern(typePattern);
    }

    @Override
    public boolean matches(Class<?> clazz) {
   
        Assert.assertNotNull(this.aspectJTypePatternMatcher);
        return this.aspectJTypePatternMatcher.matches(clazz);
    }

    public String getTypePattern() {
   
        return typePattern;
    }

    public void setTypePattern(String typePattern) {
   
        Assert.assertNotNull("匹配类型不能为空", typePattern);
        this.typePattern = typePattern;
        this.aspectJTypePatternMatcher = PointcutParser.getPointcutParserSupportingAllPrimitivesAndUsingContextClassloaderForResolution()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

穷儒公羊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值