【Springboot系列】Springboot接管所有Controller,magic-api源码阅读
创始人
2024-05-31 20:35:18
0

系列文章地址:Spring Boot学习大纲,可以留言自己想了解的技术点

最近在项目中使用了一个第三方的包 magic-api,节省了很多的时间,整体来说就是只用写sql就好了,不用写service,controller那些,全部统一处理了。

具体的使用大家可以搜索下,网上到处都是,建议去官网看。

为了使用这个包,必须得大概了解下是怎么回事,要不然用的不踏实,这里面最重要的其实是脚本执行引擎的重定义,这部分我还没有深究,

这篇文章主要关注的是如何接管所有的controller。

1、magic-api的使用流程

magic-api 在启动之后会有一个网页界面,在界面中配置开放的接口和执行的脚本

这些接口会存到数据库或者文件中,在程序运行的过程中会加载所有的接口到内存中,等待调用

2、controller主要流程

在正常的请求过程中,都知道使用注解@Controller加上RequestMapping就可以将一个方法作为接口开放出来,可以看到下面的流程

DispatchServlet做为分发器,从handlerMapping类中查找对应的Method 方法,然后注入请求对象,进行调用,

magic-api 这里接管所有的controller必然要在handlerMapping中做点手脚

3、magic-api的处理方式

1.通过magic-api-spring-boot-starter的spring.factories可以看到启动类

org.springframework.boot.autoconfigure.EnableAutoConfiguration=
org.ssssssss.magicapi.spring.boot.starter.MagicAPIAutoConfiguration

/*** magic-api自动配置类*/
@Configuration
@ConditionalOnClass({RequestMappingHandlerMapping.class})
@EnableConfigurationProperties(MagicAPIProperties.class)
@Import({MagicJsonAutoConfiguration.class, ApplicationUriPrinter.class, MagicModuleConfiguration.class, MagicDynamicRegistryConfiguration.class})
@EnableWebSocket
@AutoConfigureAfter(MagicPluginConfiguration.class)
public class MagicAPIAutoConfiguration implements WebMvcConfigurer, WebSocketConfigurer {

2.可以看到导入了几个配置类到容器中,其他的没怎么关注,直接在MagicDynamicRegistryConfiguration中查找

@Configuration
@AutoConfigureAfter(MagicModuleConfiguration.class)
public class MagicDynamicRegistryConfiguration {@Autowired@Lazyprivate RequestMappingHandlerMapping requestMappingHandlerMapping;

可以看到这里注入了RequestMappingHandlerMapping,想必一定是在这里进行的骚操作,继续往下看

3.ApiInfoMagicResourceStorage 是配置的接口,在这里做的注入

@Bean
@ConditionalOnMissingBean
public RequestMagicDynamicRegistry magicRequestMagicDynamicRegistry(ApiInfoMagicResourceStorage apiInfoMagicResourceStorage) throws NoSuchMethodException {return new RequestMagicDynamicRegistry(apiInfoMagicResourceStorage, Mapping.create(requestMappingHandlerMapping, properties.getWeb()), properties.isAllowOverride(), properties.getPrefix());
}

4.下面具体的就不做深入展开了,等下会讲原理

最后会讲所有的请求全部映射到RequestHandler里的invoke方法

*** 测试入口、实际请求入口** @param request       HttpServletRequest* @param response      HttpServletResponse* @param pathVariables 路径变量* @param parameters    表单参数&URL参数* @return 返回请求结果* @throws Throwable 处理失败抛出的异常*/
@ResponseBody
@Valid(requireLogin = false)
public Object invoke(HttpServletRequest request, HttpServletResponse response,@PathVariable(required = false) Map pathVariables,@RequestHeader(required = false) Map defaultHeaders,@RequestParam(required = false) Map parameters) throws Throwable {

4、自己实现,接管所有的controller到RequestHandler

4.1 创建springboot项目

idea一键next,没有难度,创建一个空的springboot web项目就好

4.2 创建一个controller的处理类,这里没有加controller的注解

这里直接拷贝了magic-api的RequestHandler,去除了业务逻辑

/*** 请求入口处理*/
@Component
public class RequestHandler {/*** 测试入口、实际请求入口** @param request       HttpServletRequest* @param response      HttpServletResponse* @param pathVariables 路径变量* @param parameters    表单参数&URL参数* @return 返回请求结果* @throws Throwable 处理失败抛出的异常*/@ResponseBodypublic Object invoke(HttpServletRequest request, HttpServletResponse response,@PathVariable(required = false) Map pathVariables,@RequestHeader(required = false) Map defaultHeaders,@RequestParam(required = false) Map parameters) throws Throwable {return "香菜";}
}

4.3 注册处理类到requestMapping

@Configuration
public class MappingConfig {@Autowiredpublic void setHandlerMapping(RequestMappingHandlerMapping mapping, RequestHandler handler) throws NoSuchMethodException {RequestMappingInfo info = RequestMappingInfo.paths("/caraway").methods(RequestMethod.GET).build();//    注意这里的方法签名,要和RequestHandler保持一致Method method =  RequestHandler.class.getDeclaredMethod("invoke", HttpServletRequest.class, HttpServletResponse.class, Map.class, Map.class, Map.class);mapping.registerMapping(info, handler, method);}
}

4.4 测试

输入地址:http://localhost:16002/caraway

端口记得改成你本地的端口,因为多开了几个服务,这里的端口是16002

如果看到下面的界面,证明我们注入成功了

4.5 开发遇到的问题

上面代码写完,本以为没有问题,但是在使用的时候一直报错

搜索了下:问题就是源自springboot 2.6.0后的新特性,目前版本的解决方案是在springboot的配置文件中,以下配置,修改默认映射策略:

spring.mvc.pathmatch.matching-strategy=ANT_PATH_MATCHER

或者降低版本,如使用2.5.4版本

5、总结

magic-api 减少了很多繁琐的接口,开发速度很快。

requestMapping的动态导入controller,可以将业务接管

magic-api的脚本引擎还没研究,


推荐书

2020年9月“中国好书”、斩获第十六届文津奖图书奖的科普力作——《公式之美》火热畅销中,不管你是文科生还是理科生,它值得所有人了解与翻阅!

万物速朽,唯有公式永恒!本书通过人类最美的23个公式,用人文解析数学之美,重塑人类理性堤坝,抵挡盲信洪流,聚集日益退却的独立思考者,打破快餐式的碎片刷屏时间……用趣味解读发现科学之美,用公式智慧重构思考体系!

电商链接

京东:https://item.jd.com/12717209.html

当当:http://product.dangdang.com/29121842.html

相关内容

热门资讯

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...