Spring Boot(09)——使用SpringMVC

简介:

使用SpringMVC

使用SpringMVC最简单的方法是在pom.xml中加入spring-boot-starter-web依赖,这样Spring Boot的AutoConfiguration模块将为我们自动进行SpringMVC的配置,创建好RequestMappingHandlerAdapterRequestMappingHandlerMapping等,详情可以参考org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfigurationDelegatingWebMvcConfiguration的源码。

这个时候就可以定义如下这样一个控制器,当请求/hello/json时将返回{"key1": "value1", "key2": "value2"}这样一段JSON。当Classpath下存在jackson相关的Class时就会自动添加MappingJackson2HttpMessageConverter这样一个HttpMessageConverter。关于默认添加的HttpMessageConverter可以参考WebMvcConfigurationSupportaddDefaultHttpMessageConverters()的源码说明。

@RestController
@RequestMapping("hello")
public class HelloController {

    @GetMapping("json")
    public Object jsonResult() {
        Map<String, Object> map = new HashMap<>();
        map.put("key1", "value1");
        map.put("key2", "value2");
        return map;
    }
    
}

DispatcherServlet的自动注册将由org.springframework.boot.autoconfigure.web.servlet.DispatcherServletAutoConfiguration配置,其默认启用了Servlet的异步支持。

添加自定义的HttpMessageConverter

添加自定义的HttpMessageConverter比较方便的方法是通过org.springframework.boot.autoconfigure.http.HttpMessageConverters定义,它可以在指定使用默认的HttpMessageConverter的同时添加额外的HttpMessageConverter。下面的代码中就指定了在使用默认的HttpMessageConverter的同时添加了一个自定义的CustomHttpMessageConverter

@Configuration
public class MvcConfiguration {

@Bean
public HttpMessageConverters httpMessageConverters() {
    HttpMessageConverter<?> customHttpMessageConverter = new CustomHttpMessageConverter();
    List<HttpMessageConverter<?>> additional = new ArrayList<>();
    additional.add(customHttpMessageConverter);
    HttpMessageConverters converters = new HttpMessageConverters(true, additional);
    return converters;
}

}

HttpMessageConverters有几个重载的构造方法,使用时可以参考对应的API文档选择合适的进行使用。

Spring Boot中拥有一个HttpMessageConvertersAutoConfiguration类,其会在未定义HttpMessageConverters类型的bean时,自动注册一个HttpMessageConverters类型的bean,即会通过它来使用默认的HttpMessageConverter。此外,其会在自动注册bean容器中定义的HttpMessageConverter,所以使用默认配置时也可以把需要注册的HttpMessageConverter定义为bean容器中的一个bean。HttpMessageConvertersAutoConfiguration中拥有一个StringHttpMessageConverterConfiguration类,会在bean容器中未定义StringHttpMessageConverter类型的bean时自动定义一个,使用的字符集默认是UTF-8,可以通过在application.properties中通过spring.http.encoding.charset指定。

Converter和Formatter注册

下面的代码来自于WebMvcAutoConfigurationAdapter,从代码中可以看出Spring Boot会自动注册bean容器中定义的Converter和Formatter。

@Override
public void addFormatters(FormatterRegistry registry) {
    for (Converter<?, ?> converter : getBeansOfType(Converter.class)) {
        registry.addConverter(converter);
    }
    for (GenericConverter converter : getBeansOfType(GenericConverter.class)) {
        registry.addConverter(converter);
    }
    for (Formatter<?> formatter : getBeansOfType(Formatter.class)) {
        registry.addFormatter(formatter);
    }
}

静态资源的处理

Spring Boot默认会把Classpath下的/META-INF/resources//resources//static//public/映射为静态资源路径。静态资源的相关配置由ResourceProperties类定义,对应的配置属性前缀是spring.resources。如果不想使用默认的静态资源位置,可以通过spring.resources.static-locations属性进行自定义。如果不需要把那些路径映射为静态资源路径,则可以设置spring.resources.addMappings的值为true。静态资源默认会映射为/**,即如果在/resources下拥有一个index.html文件,则可以通过/index.html请求到。可以通过spring.mvc.static-path-pattern指定静态资源映射的路径,下面代码就指定了静态资源的映射路径为/resources/**/**对应的才是真实的静态资源的路径,所以此时如果需要请求/resources路径下的index.html文件,需要通过/resources/index.html才能请求到。

spring.mvc.static-path-pattern=/resources/**

更多配置信息可以参考ResourceProperties的源代码。

浏览器图标文件

Spring Boot默认会在配置的静态资源根路径下或者是Classpath根路径下寻找favicon.ico文件。

Content Negotiation

Spring Boot默认是不支持后缀名匹配的,即请求/report.json时是不会被映射到@GetMapping("/report")的。可以通过如下方式指定支持后缀名匹配。

spring.mvc.contentnegotiation.favor-path-extension=true

关于Content Negotiation的可选配置都定义在WebMvcProperties$Contentnegotiation.class中。可以通过spring.mvc.contentnegotiation.favor-parameter=true指定支持通过查询参数指定请求类型,默认的查询请求类型查询参数是format,需要自定义时可以通过spring.mvc.contentnegotiation.parameter-name属性来指定。

更换内置Web容器

当引入了spring-boot-starter-web后默认使用的Web容器是tomcat,Spring Boot也提供了一些其它的实现,比如jetty,当需要使用jetty实现时需要先把tomcat实现从spring-boot-starter-web中排除,再添加spring-boot-starter-jetty依赖。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <exclusions>
        <!-- Exclude the Tomcat dependency -->
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<!-- Use Jetty instead -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-jetty</artifactId>
</dependency>

不启动Web容器

默认情况下你添加了spring-boot-starter-web后Spring Boot会自动帮你启动Web容器,如果不期望启动Web容器可以配置spring.main.web-application-type=none

内置容器可配置的属性

当使用内置的Web容器时可以针对内置容器进行一些自定义的配置,这些配置将由org.springframework.boot.autoconfigure.web.ServerProperties进行接收。常用的自定义配置包括server.port指定监听端口,server.servlet.contextPath指定contextPath,还可以通过server.servlet.session.*指定session相关的定义信息。下面的配置中定义了监听端口是8081,contextPath是/app,DispatcherServlet匹配的映射路径是/web/*,session的超时时间三30分钟,当指定session的超时时间时,如果不指定时间单位,默认单位是秒。

server.port=8081
server.servlet.context-path=/app
server.servlet.path=/web
server.servlet.session.timeout=30m

更多关于内置Web容器可配置的信息可以参考org.springframework.boot.autoconfigure.web.ServerProperties的API或源码。

注册Servlet/Filter等

根据Servlet3的规范,Classpath下的@WebServlet@WebFilter@WebListener标注的Class会被Web容器自动检测到,并进行注册。但是使用Spring Boot内置的Web容器时,它们是不会自动被检测到并被注册的。有两种方式可以在使用内置的Web容器时能够让它们进行自动注册。

定义为Spring bean

Spring Bean容器中的定义的Filter/Servlet和Listener会被自动注册,Filter会拦截所有请求,而Servlet的请求路径将是bean名称,然后需要以/结尾,比如下面代码中定义的Servlet的请求路径是/hello/

@Component("hello")
public class HelloServlet extends HttpServlet {

    /**
     * 
     */
    private static final long serialVersionUID = 8345578389259773375L;

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        PrintWriter writer = resp.getWriter();
        writer.write("hello");
        writer.flush();
    }

}

如果上面的Servlet对应的bean名称是servlet/hello,则其对应的请求路径为/servlet/hello/

需要注意的是当bean容器中只定义了一个Servlet时,该Servlet的映射路径不是bean名称,而是/

通过@ServletComponentScan扫描

第二种方式是还是在Class上标注@WebServlet@WebFilter@WebListener注解,然后在某个@Configuration Class上标注@ServletComponentScan,可以通过它的value、basePackages或basePackageClasses三者之一来指定需要扫描的包,如果都不指定,默认会以标注的Class所在的包作为根包进行扫描。下面的代码中先是定义了一个Servlet,使用了@WebServlet标注,并指定了映射的路径为/servlet/test。然后在Application类上使用了@ServletComponentScan标注。Spring Boot将会扫描到TestServlet类,并把它注册为一个Servlet。

@WebServlet("/servlet/test")
public class TestServlet extends HttpServlet {

    /**
     * 
     */
    private static final long serialVersionUID = -2499437569852564816L;

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        PrintWriter writer = resp.getWriter();
        writer.write("Hello Servlet.");
        writer.flush();
    }
    

}
@SpringBootApplication
@ServletComponentScan
public class Application {

    public static void main(String[] args) {
        SpringApplication app = new SpringApplication(Application.class);
        app.setAddCommandLineProperties(false);
        app.run(args);
    }

}

如果需要使用@WebServlet的方式定义Servlet,同时又希望可以访问到Spring bean,则可以同时把Servlet定义为Spring的一个bean。下面的代码中定义的Servlet可以通过/servlet/hello访问,同时它是一个Spring bean,被注入了ApplicationContext,在访问它时会输出所有定义的bean的名称。该Servlet同时也可以通过/hello/访问到,因为作为一个HttpServlet类型的bean,它也会被注册为一个Servlet,映射路径为bean名称。

@Component("hello")
@WebServlet("/servlet/hello")
public class HelloServlet extends HttpServlet {

    @Autowired
    private ApplicationContext applicationContext;
    
    /**
     * 
     */
    private static final long serialVersionUID = 8345578389259773375L;

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        PrintWriter writer = resp.getWriter();
        writer.println("bean names: ");
        for (String name : this.applicationContext.getBeanDefinitionNames()) {
            writer.println(name);
        }
        writer.flush();
    }

}

参考文档

https://docs.spring.io/spring-boot/docs/2.0.3.RELEASE/reference/html/boot-features-developing-web-applications.html#boot-features-embedded-container-servlets-filters-listeners

(注:本文是基于Spring Boot 2.0.3所写)

目录
相关文章
|
6月前
|
前端开发 Java 测试技术
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@RequestParam
本文介绍了 `@RequestParam` 注解的使用方法及其与 `@PathVariable` 的区别。`@RequestParam` 用于从请求中获取参数值(如 GET 请求的 URL 参数或 POST 请求的表单数据),而 `@PathVariable` 用于从 URL 模板中提取参数。文章通过示例代码详细说明了 `@RequestParam` 的常用属性,如 `required` 和 `defaultValue`,并展示了如何用实体类封装大量表单参数以简化处理流程。最后,结合 Postman 测试工具验证了接口的功能。
332 0
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@RequestParam
|
6月前
|
JSON 前端开发 Java
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@RequestBody
`@RequestBody` 是 Spring 框架中的注解,用于将 HTTP 请求体中的 JSON 数据自动映射为 Java 对象。例如,前端通过 POST 请求发送包含 `username` 和 `password` 的 JSON 数据,后端可通过带有 `@RequestBody` 注解的方法参数接收并处理。此注解适用于传递复杂对象的场景,简化了数据解析过程。与表单提交不同,它主要用于接收 JSON 格式的实体数据。
501 0
|
6月前
|
前端开发 Java 微服务
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@PathVariable
`@PathVariable` 是 Spring Boot 中用于从 URL 中提取参数的注解,支持 RESTful 风格接口开发。例如,通过 `@GetMapping(&quot;/user/{id}&quot;)` 可以将 URL 中的 `{id}` 参数自动映射到方法参数中。若参数名不一致,可通过 `@PathVariable(&quot;自定义名&quot;)` 指定绑定关系。此外,还支持多参数占位符,如 `/user/{id}/{name}`,分别映射到方法中的多个参数。运行项目后,访问指定 URL 即可验证参数是否正确接收。
317 0
|
6月前
|
JSON 前端开发 Java
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@RequestMapping
@RequestMapping 是 Spring MVC 中用于请求地址映射的注解,可作用于类或方法上。类级别定义控制器父路径,方法级别进一步指定处理逻辑。常用属性包括 value(请求地址)、method(请求类型,如 GET/POST 等,默认 GET)和 produces(返回内容类型)。例如:`@RequestMapping(value = &quot;/test&quot;, produces = &quot;application/json; charset=UTF-8&quot;)`。此外,针对不同请求方式还有简化注解,如 @GetMapping、@PostMapping 等。
277 0
|
6月前
|
JSON 前端开发 Java
微服务——SpringBoot使用归纳——Spring Boot中的MVC支持——@RestController
本文主要介绍 Spring Boot 中 MVC 开发常用的几个注解及其使用方式,包括 `@RestController`、`@RequestMapping`、`@PathVariable`、`@RequestParam` 和 `@RequestBody`。其中重点讲解了 `@RestController` 注解的构成与特点:它是 `@Controller` 和 `@ResponseBody` 的结合体,适用于返回 JSON 数据的场景。文章还指出,在需要模板渲染(如 Thymeleaf)而非前后端分离的情况下,应使用 `@Controller` 而非 `@RestController`
211 0
|
2月前
|
SQL Java 数据库连接
Spring、SpringMVC 与 MyBatis 核心知识点解析
我梳理的这些内容,涵盖了 Spring、SpringMVC 和 MyBatis 的核心知识点。 在 Spring 中,我了解到 IOC 是控制反转,把对象控制权交容器;DI 是依赖注入,有三种实现方式。Bean 有五种作用域,单例 bean 的线程安全问题及自动装配方式也清晰了。事务基于数据库和 AOP,有失效场景和七种传播行为。AOP 是面向切面编程,动态代理有 JDK 和 CGLIB 两种。 SpringMVC 的 11 步执行流程我烂熟于心,还有那些常用注解的用法。 MyBatis 里,#{} 和 ${} 的区别很关键,获取主键、处理字段与属性名不匹配的方法也掌握了。多表查询、动态
112 0
|
8月前
|
XML Java 应用服务中间件
Spring Boot 两种部署到服务器的方式
本文介绍了Spring Boot项目的两种部署方式:jar包和war包。Jar包方式使用内置Tomcat,只需配置JDK 1.8及以上环境,通过`nohup java -jar`命令后台运行,并开放服务器端口即可访问。War包则需将项目打包后放入外部Tomcat的webapps目录,修改启动类继承`SpringBootServletInitializer`并调整pom.xml中的打包类型为war,最后启动Tomcat访问应用。两者各有优劣,jar包更简单便捷,而war包适合传统部署场景。需要注意的是,war包部署时,内置Tomcat的端口配置不会生效。
2160 17
Spring Boot 两种部署到服务器的方式
|
8月前
|
SQL Java 数据库连接
对Spring、SpringMVC、MyBatis框架的介绍与解释
Spring 框架提供了全面的基础设施支持,Spring MVC 专注于 Web 层的开发,而 MyBatis 则是一个高效的持久层框架。这三个框架结合使用,可以显著提升 Java 企业级应用的开发效率和质量。通过理解它们的核心特性和使用方法,开发者可以更好地构建和维护复杂的应用程序。
372 29
|
6月前
|
Java 数据库 微服务
微服务——SpringBoot使用归纳——Spring Boot中的项目属性配置——指定项目配置文件
在实际项目中,开发环境和生产环境的配置往往不同。为简化配置切换,可通过创建 `application-dev.yml` 和 `application-pro.yml` 分别管理开发与生产环境配置,如设置不同端口(8001/8002)。在 `application.yml` 中使用 `spring.profiles.active` 指定加载的配置文件,实现环境快速切换。本节还介绍了通过配置类读取参数的方法,适用于微服务场景,提升代码可维护性。课程源码可从 [Gitee](https://gitee.com/eson15/springboot_study) 下载。
231 0
|
11月前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
341 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块