实现SpringBoot底层机制

Tomcat底层启动分析+Spring容器初始化+Tomcat关联Spring容器

1.任务1-创建Tomcat,并启动

(1)创建一个Maven项目,修改pom.xml文件:我们需要自己创建Tomcat对象,因此在引入的场景启动器中排除SpringBoot内嵌的Tomcat,并引入tomcat依赖库

<!--导入SpringBoot父工程-规定写法-->
<parent>
<artifactId>spring-boot-starter-parent</artifactId>
<groupId>org.springframework.boot</groupId>
<version>2.5.3</version>
</parent>
<dependencies>
<!--导入web项目场景启动器:会自动导入和web开发相关的所有依赖[jar包]-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<!--因为我们要自己创建Tomcat对象,并启动,因为我们要先排除内嵌的spring-boot-starter-tomcat-->
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-tomcat</artifactId>
</exclusion>
</exclusions>
</dependency>
<!--指定Tomcat版本,引入tomcat依赖库
1.指定版本为8.5.75
2.如果我们引入了自己指定的tomcat,一定要记住把前面的spring-boot-starter-tomcat排除
3.否则会出现GenericServletNotFound错误-->
<dependency>
<groupId>org.apache.tomcat.embed</groupId>
<artifactId>tomcat-embed-core</artifactId>
<version>8.5.75</version>
</dependency>
</dependencies>

(2)创建LiSpringApplication.java

package com.li.lispringboot;

import org.apache.catalina.startup.Tomcat;

/**
* @author 李
* @version 1.0
*/
public class LiSpringApplication {
//创建tomcat对象,并关联spring容器,然后启动tomcat
public static void run() {
try {
//创建tomcat对象
Tomcat tomcat = new Tomcat();
//设置默认端口-9090
tomcat.setPort(9090);
//启动,就会在指定端口监听
tomcat.start();
//等待请求接入
System.out.println("======9090端口等待请求接入======");
tomcat.getServer().await();
} catch (Exception e) {
e.printStackTrace();
}
}
}

(3)LiMainApp

package com.li.lispringboot;

/**
* @author 李
* @version 1.0
*/
public class LiMainApp {
public static void main(String[] args) {
//启动LiSpringBoot项目/程序
LiSpringApplication.run();
}
}

(4)测试启动main方法,后台输出如下:


打开浏览器,访问9090端口,页面一片空白,因为这时候还没有接入其他组件。

2.任务2-创建Spring容器

(1)创建Monster.java,做一个测试bean

package com.li.lispringboot.bean;

/**
* @author 李
* @version 1.0
*/
public class Monster {
}

(2)创建HelloController.java,做一个测试Controller

package com.li.lispringboot.controller;

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; /**
* @author 李
* @version 1.0
*/
@RestController
public class HelloController {
@RequestMapping("/hello")
public String Hello() {
return "Hello,I'm HelloController!";
}
}

(3)创建LiConfig.java,作为Spring的配置文件

package com.li.lispringboot.config;

import com.li.lispringboot.bean.Monster;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration; /**
* @author 李
* @version 1.0
* 配置类-作为Spring的配置文件
* 这里有一个问题,容器怎么知道要扫描哪些包?
*/
@Configuration
@ComponentScan("com.li.lispringboot")//指定要配置类扫描哪些包
public class LiConfig {
//注入Bean-Monster对象到Spring容器
@Bean
public Monster monster() {
return new Monster();
}
}

(4)LiWebApplicationInitializer.java

package com.li.lispringboot;

import org.springframework.web.WebApplicationInitializer;
import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
import org.springframework.web.servlet.DispatcherServlet; import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRegistration; /**
* @author 李
* @version 1.0
* LiWebApplicationInitializer容器初始化类的任务:
* 1.创建spring容器
* 2.加载/关联spring容器的配置-按照注解方式
* 3.完成spring容器配置的bean的创建,依赖注入
* 4.创建前端控制器(DispatcherServlet),让其持有spring容器
* 5.这的onStartup()方法是tomcat来调用,并把ServletContext对象传入
*/
public class LiWebApplicationInitializer implements WebApplicationInitializer {
@Override
public void onStartup(ServletContext servletContext) throws ServletException {
System.out.println("onStartup ...");
//加载-Spring Web Application Configuration
AnnotationConfigWebApplicationContext ac =
new AnnotationConfigWebApplicationContext(); //在ac中注册配置类LiConfig
ac.register(LiConfig.class);
ac.refresh();//完成bean的创建和配置 /*
创建注册非常重要的前端控制器-DispatchServlet
让 DispatchServlet持有spring容器-ac
这样就可以进行映射分发
*/
DispatcherServlet dispatcherServlet = new DispatcherServlet(ac);
//返回ServletRegistration.Dynamic对象
ServletRegistration.Dynamic registration
= servletContext.addServlet("app", dispatcherServlet);
//设置前端控制器的加载顺序(这里设置为当tomcat启动时,就加载)
registration.setLoadOnStartup(1);
//设置前端控制器拦截所有请求,并进行分发处理
registration.addMapping("/");
}
}

3.任务3-将Tomcat和Spring容器关联,并启动Spring容器

(1)修改LiSpringApplication,将tomcat和Spring容器关联

package com.li.lispringboot;

import org.apache.catalina.startup.Tomcat;

/**
* @author 李
* @version 1.0
*/
public class LiSpringApplication {
//创建tomcat对象,并关联spring容器,然后启动tomcat
public static void run() {
try {
//创建tomcat对象
Tomcat tomcat = new Tomcat();
/*
1.让tomcat能够将请求转发到SpringWeb容器,因此需要关联
2."/liboot" 就是我们的项目的 application context,即原来配置tomcat时的项目名称
3."D:\\IDEA-workspace\\li-springboot" 指定项目的路径
*/
tomcat.addWebapp("/liboot", "D:\\IDEA-workspace\\li-springboot");
//设置默认端口-9090
tomcat.setPort(9090);
//启动,就会在指定端口监听
tomcat.start();
//等待请求接入
System.out.println("======9090端口等待请求接入======");
tomcat.getServer().await();
} catch (Exception e) {
e.printStackTrace();
}
}
}

(2)在LiMainApp.java中重新启动项目,在浏览器中访问测试Controller,访问成功:

3.1注意事项和细节

如果启动时报异常,如下:

严重: Servlet [jsp] in web application [/liboot] threw load() exception java.lang.ClassNotFoundException: org.apache.jasper.servlet.JspServlet

解决方案是:引入对应版本的 Jasper包即可。

<dependency>
<groupId>org.apache.tomcat</groupId>
<artifactId>tomcat-jasper</artifactId>
<version>8.5.75</version>
</dependency>

day04-实现SpringBoot底层机制的更多相关文章

  1. [转]STL 容器一些底层机制

    1.vector 容器 vector 的数据安排以及操作方式,与 array 非常相似.两者的唯一区别在于空间的运用的灵活性.array 是静态空间,一旦配置了就不能改变,vector 是动态数组.在 ...

  2. C++ STL容器底层机制

    1.vector容器 vector的数据安排以及操作方式,与array非常相似.两者的唯一区别在于空间的运用的灵活性.array是静态空间,一旦配置了就不能改变.vector是动态空间,随着元素的加入 ...

  3. 探索C++的底层机制

    探索C++的底层机制 在看这篇文章之前,请你先要明白一点:那就是c++为我们所提供的各种存取控制仅仅是在编译阶段给我们的限制,也就是说是编译器确保了你在完成任务之前的正确行为,如果你的行为不正确,那么 ...

  4. tensorflow入门教程和底层机制简单解说——本质就是图计算,自动寻找依赖,想想spark机制就明白了

    简介 本章的目的是让你了解和运行 TensorFlow! 在开始之前, 让我们先看一段使用 Python API 撰写的 TensorFlow 示例代码, 让你对将要学习的内容有初步的印象. 这段很短 ...

  5. 20191031:Python底层机制

    20191031:Python底层机制 python底层从3个方面来说,分别是: 引用计数机制 垃圾回收机制 内存池机制 引用计数机制 使用引用计数来追踪内存中的对象,所有对象都有引用计数,并且这个引 ...

  6. php-浅谈php底层机制

    php-浅谈php底层机制 1. PHP的设计理念及特点 多进程模型:由于PHP是多进程模型,不同请求间互不干涉,这样保证了一个请求挂掉不会对全盘服务造成影响,当然,随着时代发展,PHP也早已支持多线 ...

  7. 深入理解SpringBoot核心机制《spring-boot-starter》

    深入理解SpringBoot核心机制<spring-boot-starter> 前言: 对于这几年java火爆天的springBoot我相信大家都有所使用过,在springBoot的项目中 ...

  8. day12-实现Spring底层机制-02

    实现Spring底层机制-02 3.实现任务阶段1 3.1知识拓展-类加载器 Java的类加载器有三种: Bootstrap类加载器 ----- 对应路径 jre/lib Ext类加载器 ----- ...

  9. day13-实现Spring底层机制-03

    实现Spring底层机制-03 7.实现任务阶段5 7.1分析 阶段5目标:bean后置处理器的实现 7.2代码实现 新增: 1.创建 InitializingBean 接口,实现该接口的 Bean ...

  10. day05-SpringMVC底层机制简单实现-01

    SpringMVC底层机制简单实现-01 主要完成:核心分发控制器+Controller和Service注入容器+对象自动装配+控制器方法获取参数+视图解析+返回JSON格式数据 1.搭建开发环境 创 ...

随机推荐

  1. Matter开发,看这一篇就够了

    1. Matter介绍 Matter(以前称为 Project Connected Home over IP 或 Project CHIP)是由CSA联盟制定的一个应用层面的标准,旨在打造一个统一的智 ...

  2. npm i -D和-s及-g以及--save的那些事

      i 是 install 的简写 -S 就是 --save 的简写 -D 就是 --save-dev 的简写 npm i module_name -S = > npm install modu ...

  3. 到什么程度才叫精通 Linux?

    大家好,我是陶朱公Boy,一个认真生活,总想超越自己的程序员. 前言 知乎上有一个提问:到什么程度才叫精通 Linux?                              ↓↓↓ 今天,我们就 ...

  4. JS 判断对象属性是否存在,判断是否包含某个属性,是否为自身属性

    壹 ❀ 引 看过博主JS 疫情宅在家,学习不能停,七千字长文助你彻底弄懂原型与原型链这篇文章的同学应该知道,文中有专门介绍这个问题.那么为什么我要另起一篇再说一次呢?原因有两个,一是介绍原型与原型链的 ...

  5. SSD寻址单元IU对寿命的影响——古猫先生存储随笔转载

    SSD寻址单元IU对寿命的影响有多大? 原创 古猫先生 存储随笔 2024-01-30 08:30 发表于浙江 [转载]SSD寻址单元IU对寿命的影响有多大? (qq.com) 随着存储技术的不断进步 ...

  6. 欧拉公式 Euler's Formula

    欧拉公式是数学中最重要的公式之一, 它涉及到了复数, 无理数, 三角函数, 简单优美 \(e^{i\theta} = cos(\theta) + isin(\theta)\) 欧拉公式代表的含义并不是 ...

  7. spring boot+layui分页实战

    项目用了layui,做了个简单的图书搜索页,分享出来. 喜欢的朋友给点个赞!!! 实现效果 开发步骤 1.前端页面和JS <!DOCTYPE html> <html xmlns=&q ...

  8. Java并发编程实例--4.控制线程打断

    Java提供了InterruptedException异常,当我们检测到线程被打断时可以抛出并在run()方法中进行捕捉. 本例中,我们将开发一个程序以实现根据文件名称在指定文件夹(包括其子目录)中搜 ...

  9. git回退至指定版本,并更新远程仓库

    1. git log   查到commit记录 2.复制 commit 后面的id 3. git reset --hard  commit 后面的id   // 回退 4. 强制更新远程仓库  git ...

  10. 第一百一十二篇: JS数组Array(一)数组基本用法

    好家伙,   1.数组 Array应该就是ECMAScript中最常用的类型了.ECMAScript数组跟其他编程语言的数组有很大区别. 跟其他语言中的数组一样,ECMAScript 数组也是一组有序 ...