前提
这篇文章是《SpringBoot2.x入门》专辑的「第3篇」文章,使用的SpringBoot
版本为2.3.1.RELEASE
,JDK
版本为1.8
。
主要介绍SpringBoot
的web
模块引入,会相对详细地分析不同的Servlet
容器(如Tomcat
、Jetty
等)的切换,以及该模块提供的SpringMVC
相关功能的使用。
依赖引入
笔者新建了一个多模块的Maven
项目,这次的示例是子模块ch1-web-module
。
SpringBoot
的web
模块实际上就是spring-boot-starter-web
组件(下称web
模块),前面的文章介绍过使用BOM
全局管理版本,可以在(父)POM
文件中添加dependencyManagement
元素:
<properties> <spring.boot.version>2.3.1.RELEASE</spring.boot.version> </properties> <dependencyManagement> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>${spring.boot.version}</version> <scope>import</scope> <type>pom</type> </dependency> </dependencies> </dependencyManagement> 复制代码
接来下在(子)POM
文件中的dependencies
元素引入spring-boot-starter-web
的依赖:
<dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> 复制代码
项目的子POM
大致如下:
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>club.throwable</groupId> <artifactId>spring-boot-guide</artifactId> <version>1.0-SNAPSHOT</version> </parent> <artifactId>ch1-web-module</artifactId> <version>1.0-SNAPSHOT</version> <packaging>jar</packaging> <name>ch1-web-module</name> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> <build> <finalName>ch1-web-module</finalName> <!-- 引入spring-boot-maven-plugin以便项目打包 --> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <version>${spring.boot.version}</version> <executions> <execution> <goals> <goal>repackage</goal> </goals> </execution> </executions> </plugin> </plugins> </build> </project> 复制代码
spring-boot-starter-web
模块中默认使用的Servlet
容器是嵌入式(Embedded
)Tomcat
,配合打包成一个Jar
包以便可以直接使用java -jar xxx.jar
命令启动。
SpringMVC的常用注解
web
模块集成和扩展了SpringMVC
的功能,移除但兼容相对臃肿的XML
配置,这里简单列举几个常用的Spring
或者SpringMVC
提供的注解,简单描述各个注解的功能:
「组件注解:」
@Component
:标记一个类为Spring
组件,扫描阶段注册到IOC
容器。@Repository
:标记一个类为Repository
(仓库)组件,它的元注解为@Component
,一般用于DAO
层。@Service
:标记一个类为Service
(服务)组件,它的元注解为@Component
。@Controller
:标记一个类为Controller
(控制器)组件,它的元注解为@Component
,一般控制器是访问的入口,衍生注解@RestController
,简单理解为@Controller
标记的控制器内所有方法都加上下面提到的@ResponseBody
。
「参数注解:」
@RequestMapping
:设置映射参数,包括请求方法、请求的路径、接收或者响应的内容类型等等,衍生注解为GetMapping
、PostMapping
、PutMapping
、DeleteMapping
、PatchMapping
。@RequestParam
:声明一个方法参数绑定到一个请求参数。@RequestBody
:声明一个方法参数绑定到请求体,常用于内容类型为application/json
的请求体接收。@ResponseBody
:声明一个方法返回值绑定到响应体。@PathVariable
:声明一个方法参数绑定到一个URI
模板变量,用于提取当前请求URI
中的部分到方法参数中。
编写控制器和启动类
在项目中编写一个控制器club.throwable.ch1.controller.HelloController
如下:
import lombok.extern.slf4j.Slf4j; import org.springframework.http.ResponseEntity; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import java.util.Optional; @Slf4j @Controller @RequestMapping(path = "/ch1") public class HelloController { @RequestMapping(path = "/hello") public ResponseEntity<String> hello(@RequestParam(name = "name") String name) { String value = String.format("[%s] say hello", name); log.info("调用[/hello]接口,参数:{},响应结果:{}", name, value); return ResponseEntity.of(Optional.of(value)); } } 复制代码
HelloController
只提供了一个接收GET
请求且请求的路径为/ch1/hello
的方法,它接收一个名称为name
的参数(参数必传),然后返回简单的文本:${name} say hello
。可以使用衍生注解简化如下:
import lombok.extern.slf4j.Slf4j; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; import java.util.Optional; @Slf4j @RestController @RequestMapping(path = "/ch1") public class HelloController { @GetMapping(path = "/hello") public ResponseEntity<String> hello(@RequestParam(name = "name") String name) { String value = String.format("[%s] say hello", name); log.info("调用[/hello]接口,参数:{},响应结果:{}", name, value); return ResponseEntity.of(Optional.of(value)); } } 复制代码
接着编写一个启动类club.throwable.ch1.Ch1Application
,启动类是SpringBoot
应用程序的入口,需要提供一个main
方法:
package club.throwable.ch1; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class Ch1Application { public static void main(String[] args) { SpringApplication.run(Ch1Application.class, args); } } 复制代码
然后以DEBUG
模式启动一下:
Tomcat
默认的启动端口是8080
,启动完毕后见日志如下:
用用浏览器访问http://localhost:8080/ch1/hello?name=thrwoable
可见输出如下:
至此,一个简单的基于spring-boot-starter-web
开发的web
应用已经完成。
切换Servlet容器
有些时候由于项目需要、运维规范或者个人喜好,并不一定强制要求使用Tomcat
作为Servlet
容器,常见的其他选择有Jetty
、Undertow
,甚至Netty
等。以Jetty
和Undertow
为例,切换为其他嵌入式Servlet
容器需要从spring-boot-starter-web
中排除Tomcat
的依赖,然后引入对应的Servlet
容器封装好的starter
。
「切换为Jetty
」,修改POM
文件中的dependencies
元素:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring‐boot‐starter‐jetty</artifactId> </dependency> 复制代码
「切换为Undertow
」,修改POM
文件中的dependencies
元素:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring‐boot‐starter‐undertow</artifactId> </dependency> 复制代码
小结
这篇文章主要分析了如何基于SpringBoot
搭建一个入门的web
服务,还简单介绍了一些常用的SpringMVC
注解的功能,最后讲解如何基于spring-boot-starter-web
切换底层的Servlet
容器。学会搭建MVC
应用后,就可以着手尝试不同的请求方法或者参数,尝试常用注解的功能。
代码仓库
这里给出本文搭建的web
模块的SpringBoot
应用的仓库地址(持续更新):
(本文完 c-2-d e-a-20200703 23:09 PM)