java 相关的后端框架接触较深的有 Jfinal 和 SpringBoot。Jfinal 因为是国产的框架,当时心里也是有股执念在,就大致过了一遍源码,后续的项目中只要是没有技术栈限制,一般我都会用 Jfinal ,但渐渐发现生态相比 SpringBoot 还是有差距的,为了缩短项目开发周期,继而转变战略去研究了一下 SpringBoot 。生态真的很重要啊!这真不是技术的高低问题。
没接触 SpringBoot 的时候,其实 Spring 玩的还行,但是还是想折腾折腾。了解,熟练,精通,不看看源码怎么能算精通呢???
一个请求怎么从 Tomcat 转发到 SpringBoot 中呢,一个请求又是怎么在 SpringBoot 中处理的呢?翻翻源码吧,不难,但是了解了对你想扩展和开发用处很大。比较喜欢用 SpringBoot 自带的工具类 。
源码只大致讲下逻辑,毕竟不是写框架,没必要逐字逐句的看
Tomcat 会根据请求找到对应的 Servlet ,然后根据 Servlet 和 Request 生成对应 FilterChain ,然后责任链模式执行 Filter 和 Servlet 达到处理请求的逻辑。
// 分配servlet实例以处理此请求
if (!unavailable) {
servlet = wrapper.allocate();
}
复制代码
@Slf4j
@WebServlet(name = "MyServletJson", urlPatterns = "/servlet/json", description = "Servlet的说明")
@Component
public class MyServletJson extends HttpServlet {
}
复制代码
请求 url 为 /servlet/json 时, servlet = wrapper.allocate();
分配的为 MyServletJson 实例
url 没有精准匹配到注册的 Servlet , servlet = wrapper.allocate();
分配的为 DispatcherServlet 实例
// 为当前请求创建 过滤器链
ApplicationFilterChain filterChain =
ApplicationFilterFactory.createFilterChain(request, wrapper, servlet);
复制代码
执行这个方法,会责任链模式执行过滤器和对应的 Servlet
filterChain.doFilter(request.getRequest(), response.getResponse());
ApplicationFilterChain.doFilter(ServletRequest request, ServletResponse response){
internalDoFilter(request,response){
servlet.service(request, response);
}
}
以上代码流程简单展示了请求怎么从 Tomcat 到 SpringBoot 框架中,简单看下代码,了解流程即可,我们的重心应该放到 SpringBoot 中。
Tomcat 中servlet.service(request, response);
将请求转发到了 Servlet(DispatcherServlet)
中
Idea 中查看继承关系,我们主要关心的是 HttpServlet , FrameworkServlet, DispatcherServlet。
debug 即可看到代码走向,最终的处理逻辑集中在 DispatcherServlet.doDispatch
中
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
// 确定当前请求的处理器执行链(确定当前 request 执行那些拦截器)
HandlerExecutionChain mappedHandler=getHandler(processedRequest);
// 根据 handler 确定当前请求的处理器适配器。
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
// 执行拦截器中的 preHandle 方法,责任链执行,有一个执行返回 false ,执行对应的 afterHandle 链结束
if (!mappedHandler.applyPreHandle(processedRequest, response)) {
return;
}
// 执行 aop 和 controller,service,dao,
// restful接口时,在这个逻辑通过jackson 将数据写入到 response并且返回的 mv=null;
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
// 设置 ModelAndView 中的默认 View
applyDefaultViewName(processedRequest, mv);
// 链式执行拦截器中 postHandle
mappedHandler.applyPostHandle(processedRequest, response, mv);
// View.render 将数据写入到 Response 中返回
processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
// 链式执行拦截器中 afterHandle
mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
}
图也不行画了,直接网上找了一个???
视图渲染部分(View.render),我们也可以封装自己 View。比如常用的文件下载,我们可以封装自己 View ,只需要传入对应文件流和文件名称即可达到下载的目的
HandlerExecutionChain mappedHandler=getHandler(processedRequest);
常用的 HandlerMapping有:
处理 restful 接口,对应的handler 为 HandlerMethod
使用 RequestMappingHandlerMapping 我们可以拿到 url 对应的 HandlerMethod,HandlerMethod 可以获得方法和类上的注解可以进行基础接口权限数据的初始化,从此告别手动补数据
转发或重定向链接,在此配置的WebMvcConfigurer.addViewControllers();
handler 为ParameterizableViewController
处理页面或者静态资源,handler 为 ResourceHttpRequestHandler
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
常用的 HandlerAdapter 有:
处理 restful 接口,handler 为 HandlerMethod
处理简单的 url 转发或重定向,WebMvcConfigurer.addViewControllers();
handler 为ParameterizableViewController
处理静态资源,handler 为 ResourceHttpRequestHandler
------本文完结-------
感谢你的阅读,如果喜欢的话评论、转发一下再走吧!!!
以后会有更多精彩内容呈现欢迎关注!!!!