手写@MapperScan
- 定义一个EnableMapperScan注解
@Import(MapperProxyHandlerRegister.class) 标注将MapperProxyHandlerRegister导入到容器中。
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Import(MapperProxyHandlerRegister.class)
public @interface EnableMapperScan {String baskPackages();}
- MapperProxyHandlerRegister
ImportBeanDefinitionRegistrar 是一个特殊组件,实现了这个接口的组件registry会动态给容器中批量注册组件,这里的关键就是获取到baskPackages 指定的路径,然后批量扫描classpath下的所有的接口,然后定义BeanDefinition,注册到容器中。
注意:AnnotationMetadata metadata封装的是标注了@Import注解的那个类。这样就可以拿到最终因哪个类扫描的这个配置类。
/*** Mapper组件注册器*/
public class MapperProxyHandlerRegister implements ImportBeanDefinitionRegistrar {/*** @param metadata 注意:metadata是Spring解析@Import注解标注的那个类的元信息,而不是导入的配置类的元信息* @param registry BeanDefinition注册器* @param generator BeanName生成器*/@Overridepublic void registerBeanDefinitions(AnnotationMetadata metadata,BeanDefinitionRegistry registry,BeanNameGenerator generator) {//这里的这个metadata封装的是标注了@Import注解的那个类,如果标注@Import注解因其他类似@Enablexxx注解进一步解析到的//那这里的metadata封装的就是解析@Enablexxx那个类的元数据信息,//总之,metadata封装的就是Spring原始解析类上关联的@Import注解的类Map<String, Object> annotationAttributes = metadata.getAnnotationAttributes(EnableMapperScan.class.getName());String baskPackages = annotationAttributes.get("baskPackages").toString();//获取此包下所有的类try {Set<Class<?>> scan = scan(baskPackages);if (CollUtil.isNotEmpty(scan)){//遍历接口for (Class<?> mapperClass : scan) {//注册到容器RootBeanDefinition definition = new RootBeanDefinition();definition.setBeanClass(MapperFactoryBean.class);//设置构造方法参数值ConstructorArgumentValues constructorArgumentValues = new ConstructorArgumentValues();constructorArgumentValues.addIndexedArgumentValue(0,mapperClass);definition.setConstructorArgumentValues(constructorArgumentValues);//definition.setFactoryMethodName("getObject");//注册Beanregistry.registerBeanDefinition(mapperClass.getName(),definition);}}} catch (Exception e) {e.printStackTrace();}}/*1. 创建A --> 依赖B --> 创建B(依赖A) -> 为A创建代理对象 ,并从缓存中拿,后注入A的是一个代理对象。* */public static Set<Class<?>> scan(String basePackage) throws IOException, ClassNotFoundException {// 创建一个扫描器//ClassPathScanningCandidateComponentProvider scanner = new ClassPathScanningCandidateComponentProvider(false);// 可以添加过滤器来筛选特定类型的类,这里先不添加任何过滤器,获取所有类// 例如,如果只想获取带有特定注解的类,可以添加如下代码// scanner.addIncludeFilter(new AnnotationTypeFilter(YourAnnotation.class));// 如果只想获取特定接口的实现类,可以添加如下代码// scanner.addIncludeFilter(new AssignableTypeFilter(YourInterface.class));// 构建要扫描的资源路径模式ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +resolveBasePackage(basePackage) + "/**/*.class";Resource[] resources = resolver.getResources(packageSearchPath);MetadataReaderFactory metadataReaderFactory = new SimpleMetadataReaderFactory();Set<Class<?>> classes = new HashSet<>();for (Resource resource : resources) {MetadataReader metadataReader = metadataReaderFactory.getMetadataReader(resource);String sourceClassName = metadataReader.getClassMetadata().getClassName();if (metadataReader.getClassMetadata().isInterface()){//只扫描mapper接口classes.add(Class.forName(sourceClassName));}//System.out.println("sourceClassName = " + sourceClassName);// 排除接口和抽象类,只获取具体类//if (!metadataReader.getClassMetadata().isInterface() &&!metadataReader.getClassMetadata().isAbstract()) {// String className = metadataReader.getClassMetadata().getClassName();// classes.add(Class.forName(className));//}}return classes;}private static String resolveBasePackage(String basePackage) {return basePackage.replace('.', '/');}
- MapperProxy
这个类封装了代理对象的核心方法,主要就是获取mapper接口上标注了@Select @Update @Delete注解的方法,解析sql参数,获取sqlsessionfactory,然后执行sql。这里简单模拟输出获取到的sql。
/**mapper组件动态代理方法拦截实现*/
public class MapperProxy implements InvocationHandler {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {if (method.getDeclaringClass() == Object.class){return method.invoke(this,args);}String name = method.getDeclaringClass().getName();String methodName = method.getName();System.out.println("mapperProxy 执行方法,被代理对象拦截 ===》 " + name + "." +methodName + "()");Select select = method.getAnnotation(Select.class);if (select != null){String[] value = select.value();//拦截方法System.out.println("拦截到的SQL:===> " + Arrays.toString(value));}Class<?> returnType = method.getReturnType();if (Number.class.isAssignableFrom(returnType) || returnType.isPrimitive()){return 1;}return null;}
}
- MapperFactoryBean
这个类实现了FactoryBean 接口,实现了这个接口的类Spring会进一步调用getObject方法创建Bean对象,这个Bean对象的类型是getObjectType返回的数据类型,由此,这个类型就是真正的Mapper接口类型,可以实现基于Mapper接口的自动装配。
public class MapperFactoryBean implements FactoryBean {private Class mapperInterFaceClazz;public MapperFactoryBean(Class mapperInterFaceClazz){this.mapperInterFaceClazz = mapperInterFaceClazz;}@Overridepublic Object getObject() throws Exception {/**在创建代理对象时,传入需要实现的接口mapperInterFaceClazz如果这个接口还继承了其他接口,在基于实现类调方法时,父接口的方法也会被代理对象拦截。原理很简单,面向对象三大特征之一: 继承,子类继承父类,就把父类所有的内容全都继承JVM在判断时,如果满足继承结构以及安全检查*/return Proxy.newProxyInstance(mapperInterFaceClazz.getClassLoader(),new Class[]{mapperInterFaceClazz},new MapperProxy());}@Overridepublic Class<?> getObjectType() {return mapperInterFaceClazz;}}
- MainApp
@SpringBootApplication
@EnableMapperScan(baskPackages = "com.example.ssm3.mapper")
相关文章:
手写@MapperScan
定义一个EnableMapperScan注解 Import(MapperProxyHandlerRegister.class) 标注将MapperProxyHandlerRegister导入到容器中。 Target({ElementType.METHOD,ElementType.TYPE}) Retention(RetentionPolicy.RUNTIME) Import(MapperProxyHandlerRegister.class) public interface…...
【C++】深入理解迭代器(Iterator)
博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 💯前言💯什么是迭代器?迭代器与指针的比较 💯std::string 中的迭代器示例代码与图示分析运行结果:图示说明: 小提示 💯正…...
后端Java开发:第十二天
第十二天:封装 - 理解与应用 欢迎来到今天的学习内容!今天,我们将一起深入探讨 Java 中的 封装(Encapsulation)。封装是面向对象编程的四大基本特性之一,它的核心思想是把对象的状态(属性&…...
记录一下Coding一直不能clone
配置 下载git客户端,进行配置 git config --list user.name姓名全称 user.emailIAM_xxxxxx.com ,这个就是你的邮箱地址 user.signingkey 注册coding平台的密码 一般不需要配置公钥私钥 下载TortoiseGit,配置这几个参数 配置凭据管理器 注意 这里用户名是…...
LLM加速方法,Adapter Tuning和Prompt Tuning的区别及原理举例
LLM加速方法 目录 LLM加速方法整体结构各分支内容Adapter Tuning和Prompt Tuning的区别及原理举例一、区别二、总结整体结构 基于模型微调、基于模型压缩、基于分布式并行处理。 各分支内容 基于模型微调: 包含Adapter Tuning、Prompt Tuning、LoRA三个子类别。这些技术主要…...
【SVN】版本发布快捷操作
摘要:因为每次发版都需要制作一份相同的文件夹,而大部分的包都不需要变更,但是文件又非常大,记录自己的操作经验。 首先在SVN Repository Browser 界面把上一次的版本复制一份,复制的时候重命名为新的版本号 右击要复…...
GitLab 创建项目、删除项目
1、创建项目 点击左上角图标,回到首页 点击 Create a project 点击 Create blank project 输入项目名称,点击Create Project 创建成功 2、删除项目 进入项目列表 点击对应项目,进入项目 进入Settings页面 拖到页面底部,展开Adva…...
STM32-笔记37-吸烟室管控系统项目
一、项目需求 1. 使用 mq-2 获取环境烟雾值,并显示在 LCD1602 上; 2. 按键修改阈值,并显示在 LCD1602 上; 3. 烟雾值超过阈值时,蜂鸣器长响,风扇打开;烟雾值小于阈值时,蜂鸣器不响…...
VisionPro软件Image Stitch拼接算法
2D图像拼接的3种情景 1.一只相机取像位置固定,或者多只相机固定位置拍图,硬拷贝拼图,采用CopyRegion工具实现 2.一只或多只相机在多个位置拍照,相机视野互相重叠,基于Patmax特征定位后,无缝 拼图ÿ…...
【从零开始入门unity游戏开发之——unity篇09】unity6基础入门——Unity游戏对象和组件的本质、Unity中的反射机制
文章目录 一、Unity游戏对象和组件的本质1、开发游戏的本质2、万物之根本——空对象2.1 什么是空对象?2.2 创建空对象 3、Unity游戏对象的本质4、 transform组件4.1 transform组件的重要性4.2 修改transform的值4.4 **Transform的作用**4.3 重置transform的值 5、总…...
【Linux】深入理解文件系统(超详细)
目录 一.磁盘 1-1 磁盘、服务器、机柜、机房 📌补充: 📌通常网络中用高低电平,磁盘中用磁化方向来表示。以下是具体说明: 📌如果有一块磁盘要进行销毁该怎么办? 1-2 磁盘存储结构 编辑…...
MoEs and Transformers 笔记
ref:https://huggingface.co/blog/zh/moe#%E7%94%A8router-z-loss%E7%A8%B3%E5%AE%9A%E6%A8%A1%E5%9E%8B%E8%AE%AD%E7%BB%83 MoEs and Transformers Transformer 类模型明确表明,增加参数数量可以提高性能,因此谷歌使用 GShard 尝试将 Transformer 模型…...
在Linux中,如何禁用root用户直接SSH登录?
在Linux中禁用root用户的直接SSH登录是为了增强系统的安全性,因为允许root用户通过SSH远程登录会增加服务器被暴力破解的风险。以下是在Linux系统中禁止root用户直接SSH登录的步骤: 编辑SSH配置文件: 打开/etc/ssh/sshd_config文件ÿ…...
用Python实现简单的任务自动化
目录 1. 自动发送邮件提醒 2. 自动备份文件 3. 自动下载网页内容 总结 在现代工作和生活中,任务自动化可以极大地提高效率和准确性。Python,作为一种功能强大且易于学习的编程语言,是实现任务自动化的理想选择。本文将通过几个简单而实用的案例,展示如何用Python实现任…...
为AI聊天工具添加一个知识系统 之26 资源存储库和资源管理器
本文要点 资源存储库 为了能完成本项目(“为AI聊天工具增加一个知识系统”,其核心能力是“语言处理” ,该能力的最大挑战 当仁不让的应该是自然语言处理)的设计,我们考虑一个问题:在自然语言处理中&#…...
Windows10环境下安装RabbitMq折腾记
最近有个老项目需要迁移到windows10环境,用的是比较老的rabbitmq安装包,如下所示。经过一番折腾,死活服务起不来,最终果断放弃老版本启用新版本。现在把折腾过程记录下: 一、安装erlang 安装完成后的目录结构ÿ…...
对快速由表及里说拜拜/如何正确运用由表及里
你是不是还:看到一男子拖走一女子就以为小情侣吵架而已(可能人贩子);看到男友对你好个几次就从此死心塌地(可能有手就行,细节装装而已)结果耽误终身;看到女同事对你微笑不排斥就以为…...
spring mvc源码学习笔记之八
本文说点儿简单的。 如果你想研究基于 XML 配置的 spring mvc 的话,可以简单扫一眼本文。 在基于 XML 配置的 spring mvc 开发中,我们主要就是通过 spring 提供的各种标签来配置。 但是,大家是不是都有个疑问,spring 到底给我们提…...
探秘5网口IIOT网关
在当今这个科技飞速发展的时代,工业领域正经历着一场深刻的变革,而工业物联网网关在其中扮演着至关重要的角色。 什么是IIOT网关 工业物联网网关,简单来说,就是连接工业现场设备与云端或者上层管理系统的关键桥梁。 而明达技术研…...
左神算法基础巩固--5
文章目录 前缀树生成前缀树查询前缀树查询字符串加入过几次查询所有加入的字符串中,有几个是以pre这个字符串作为前缀 删除前缀树中的某个字符串 贪心算法解题 前缀树 生成前缀树 要想生成一棵前缀树,需要先创建一个根节点,这个根节点有26条…...
GraalVM Native Image安全性加固实战:5步完成TLS/反射/动态代理全链路可信验证,规避97.3% CVE-2023类漏洞
第一章:GraalVM Native Image安全性加固实战总览GraalVM Native Image 将 Java 应用编译为独立、零依赖的原生可执行文件,显著提升启动速度与内存效率,但其静态链接特性也引入了独特的安全挑战:反射、动态代理、JNI 和资源加载等运…...
打印机驱动怎么下载?靠谱渠道推荐,轻松解决安装难题
在日常家用、办公打印场景中,打印机驱动失效、安装失败、设备无法识别是最常见的问题。绝大多数用户的核心困扰,就是不清楚打印机驱动怎么下载,盲目在网络上搜索后,要么下载到不兼容的驱动,要么碰到捆绑软件、恶意程序…...
终极指南:如何免费解锁Cursor Pro高级功能完整教程
终极指南:如何免费解锁Cursor Pro高级功能完整教程 【免费下载链接】cursor-free-vip [Support 0.45](Multi Language 多语言)自动注册 Cursor Ai ,自动重置机器ID , 免费升级使用Pro 功能: Youve reached your trial …...
番茄小说下载器完整使用指南:免费下载保存任何小说
番茄小说下载器完整使用指南:免费下载保存任何小说 【免费下载链接】fanqienovel-downloader 下载番茄小说 项目地址: https://gitcode.com/gh_mirrors/fa/fanqienovel-downloader 番茄小说下载器是一款功能强大的开源工具,专门用于批量下载和保存…...
自动化测试新思路:OpenClaw+Qwen3-4B生成与执行单元测试用例
自动化测试新思路:OpenClawQwen3-4B生成与执行单元测试用例 1. 为什么需要AI辅助测试开发 作为一名长期奋战在一线的开发者,我深知单元测试的重要性,但编写测试用例的过程往往枯燥且耗时。特别是在面对复杂业务逻辑时,手动编写测…...
如何快速掌握DankDroneDownloader:无人机固件管理的完整指南
如何快速掌握DankDroneDownloader:无人机固件管理的完整指南 【免费下载链接】DankDroneDownloader A Custom Firmware Download Tool for DJI Drones Written in C# 项目地址: https://gitcode.com/gh_mirrors/da/DankDroneDownloader 当你想完全掌控自己的…...
3天打造个性化音乐服务:KuGouMusicApi全场景开发指南
3天打造个性化音乐服务:KuGouMusicApi全场景开发指南 【免费下载链接】KuGouMusicApi 酷狗音乐 Node.js API service 项目地址: https://gitcode.com/gh_mirrors/ku/KuGouMusicApi KuGouMusicApi是一套基于Node.js构建的酷狗音乐API服务(应用程序…...
打卡信奥刷题(3085)用C++实现信奥题 P7095 [yLOI2020] 不离
P7095 [yLOI2020] 不离 题目背景乱玄黄时序,探风林实虚。 我要你共我奇谈怪趣。 任日月斑斓,待春秋兴残。 我要我们有聚无散。——银临《不离》 题目描述 这道题目来自 zxy 哔哔,咕咕让哔哔选一首歌作为题目名,但是哔哔说没有想好…...
如何快速掌握TensorFlow模块化架构:开发者终极指南
如何快速掌握TensorFlow模块化架构:开发者终极指南 【免费下载链接】community Stores documents used by the TensorFlow developer community 项目地址: https://gitcode.com/gh_mirrors/community1/community TensorFlow作为全球最流行的机器学习框架&…...
AppFlowy 终极安装配置完整教程:快速搭建个人AI知识库
AppFlowy 终极安装配置完整教程:快速搭建个人AI知识库 【免费下载链接】AppFlowy Bring projects, wikis, and teams together with AI. AppFlowy is the AI collaborative workspace where you achieve more without losing control of your data. The leading ope…...
