Spring基础系列-Web开发
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9996902.html
SpringBoot基础系列-web开发
概述
web开发就是集成Spring MVC进行开发,非REST开发。
整合Spring MVC
Spring MVC自动配置
当我们在POM中添加spring-boot-starter-web之后,SpringBoot就会自动进行SpringMVC整合配置,这些配置内容包括:
- 自动创建ContentNegotiatingViewResolver和BeanNameViewResolver的实例Bean
- 提供对持静态资源,包括WebJar的支持
- 自动创建Converter、GenericConverter和Formatter的实例Bean
- 提供对HttpMessageConverters的支持
- 自动创建MessageCodesResolver实例Bean
- 提供对静态欢迎页面index.html的支持
- 定制Favicon的支持
- 自动使用ConfigurableWebBindingInitializer实例Bean
定制Spring MVC
定制方式一
保留默认的自动配置,然后在其基础上新增一些配置:
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
// 添加针对swagger的处理,避免swagger404
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler("swagger-ui.html")
.addResourceLocations("classpath:/META-INF/resources/");
}
//...自定义实现WebMvcConfigurer中的若干默认方法
}
定制方式二
完全控制Spring MVC,手动定制其各种功能:
@EnableWebMvc
@Configuration
public class WebMvcConfig {
//...自定义实现WebMvcConfigurer中的若干默认方法
}
定制方式三
定制RequestMappingHandlerMapping、RequestMappingHandlerAdapter或ExceptionHandlerExceptionResolver实例:
通过声明一个WebMvcRegistrationsAdapter实例来提供这些组件。
HttpMessageConverters
即Http消息转换器,主要用于转换Http请求和响应,比如Objects会被自动转换成为JSON格式或者XML格式。编码类型默认为UTF-8。
可以定制该转换器,方式为:
@Configuration
public class MyConfiguration {
// 定制HttpMessageConverters
@Bean
public HttpMessageConverters customConverters() {
HttpMessageConverter<?> additional = ...
HttpMessageConverter<?> another = ...
return new HttpMessageConverters(additional, another);
}
}
定制JSON序列化与反序列化
SpringBoot默认使用Jackson进行Json操作。
可以定制序列化与反序列化操作,方式为:
@JsonComponent
public class Example {
public static class Serializer extends JsonSerializer<SomeObject> {
// 定制json序列化逻辑...
}
public static class Deserializer extends JsonDeserializer<SomeObject> {
// 定制json反序列化逻辑...
}
}
或者
@JsonComponent
public static class Serializer extends JsonSerializer<SomeObject> {
// 定制json序列化逻辑...
}
关于注解@JsonComponent
看看源码:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface JsonComponent {
@AliasFor(annotation = Component.class)
String value() default "";
}
可以看到该注解是一个@Component,那么他的作用就类似与@Component,主要用于注册Bean。
MessageCodesResolver
即消息编码解析器,是Spring MVC内部用来生成错误编码来表示错误信息的。
静态内容
[待补充]
欢迎页面
SpringBoot首先会查找index.html静态欢迎页面,如果找不到再查找index.ftl之类的模板欢迎页面。
定制应用图标
SpringBoot会在配置的静态资源路径和类路径中(先后顺序)查找favicon.ico图标,将其用作应用图标。
ConfigurableWebBindingInitializer
SpringMVC通过一个WebBindingInitializer来为特定的请求提供一个WebDataBinder。如果自定义了ConfigurableWebBindingInitializer,那么SpringBoot将自动配置使SpringMVC使用它。
模板引擎
SpringBoot提供对以下模板引擎的自动支持:
- Freemarker
- Groovy
- Thymeleaf
- Mustache
错误处理
默认情况下,Spring Boot提供了一个/error映射,以合理的方式处理所有错误,并在servlet容器中注册为“全局”错误页面。
即在SpringBoot内部提供了这么一个控制器类BasicErrorController,接收/error请求,然后针对浏览器请求和客户端请求两种情况作了映射,分别返回不同的内容。浏览器请求返回一个公共的错误页面,而客户端请求则返回一个ResponseEntity实例。
定制错误处理功能
方式一:定制错误页面
定制错误页面就是针对不同的code定义页面
在resources目录下的static目录(或者templates目录)下定义error目录,在error目录中定义401.html,404.html,500.html等错误页面,一旦SpringBoot应用发生了401、404、500错误就会跳转到自定义的错误页面中,而对于未自定义编码的错误还会跳转到公共错误页面
/static/error/404.html
/static/error/500.html
/templates/error/404.ftl
/templates/error/500.ftl
注意:必须定义到上面所说的目录中,而且名称必须为:错误编码.html格式,如果不按照以上规则,则定制不成功,其实如果想要自定义错误页面地址和名称也是可以的,只不过需要多加一个步骤:
添加EmbeddedServletContainerCustomizer的Bean实例用于手动设置错误页面的映射关系:
假如将500.html错误页面创建到resources目录下,也就是类路径根目录下,那么就需要使用如下自定义的ErrorViewResolver来处理了:
/500.html
内容为:<p>根目录的500错误文件</p>
MyErrorVivwResolver.java
@Component
public class MyErrorVivwResolver implements ErrorViewResolver,ApplicationContextAware {
@Override
public ModelAndView resolveErrorView(HttpServletRequest request, HttpStatus status, Map<String, Object> model) {
Resource resource = this.applicationContext.getResource("classpath:/");
try {
resource = resource.createRelative(status.value() + ".html");
} catch (IOException e) {
e.printStackTrace();
}
ModelAndView modelAndView = new ModelAndView(new HtmlResourceView(resource), model);
return modelAndView;
}
ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
private static class HtmlResourceView implements View {
private Resource resource;
HtmlResourceView(Resource resource) {
this.resource = resource;
}
@Override
public String getContentType() {
return MediaType.TEXT_HTML_VALUE;
}
@Override
public void render(Map<String, ?> model, HttpServletRequest request,
HttpServletResponse response) throws Exception {
response.setContentType(getContentType());
FileCopyUtils.copy(this.resource.getInputStream(),
response.getOutputStream());
}
}
}
代码中不少内容是抄自SpringBoot内置的DefaultErrorViewResolver。
页面请求:
http://localhost:8080/error
页面跳转到500错误页面,页面显示:
根目录的500错误文件
方式二:无SpringMVC的错误页面映射(一般不涉及)
在不使用SpringMVC的情况下进行错误页面映射,需要使用ErrorPageRegistrar(错误页面注册器)来直接注册ErrorPages(错误页面)。
这个注册器直接与底层嵌入式servlet容器一起工作,即使没有Spring MVC的DispatcherServlet也可以工作。
跨域请求
跨源资源共享(Cross-origin resource sharing, CORS)是由大多数浏览器实现的W3C规范,它允许您以灵活的方式指定哪种跨域请求被授权,而不是使用一些不太安全、功能不太强大的方法,比如IFRAME或JSONP。
有两种配置方式:
全局配置
全局配置针对的是应用的所有控制器接口
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
// 跨域请求全局配置
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/book/**");
}
//...自定义实现WebMvcConfigurer中的若干默认方法
}
细粒度配置
细粒度指的是针对单个控制器中的方法,甚至是单个方法进行配置,使用@CrossOrigin注解
@RestController
@RequestMapping("/book")
@Api(description = "书籍接口")
@Log4j2
@CrossOrigin(maxAge = 3600)
public class BookApi {
@Autowired
private BookService bookService;
@CrossOrigin("http://localhost:8081")
@RequestMapping(value = "/getBook", method = RequestMethod.GET)
@ApiOperation(value = "获取一本书籍", notes = "根据ID获取书籍", httpMethod = "GET")
public ResponseEntity<Book> getBook(final int bookId){
return bookService.getBook(bookId);
}
}
Spring基础系列-Web开发的更多相关文章
- Spring基础系列-Spring事务不生效的问题与循环依赖问题
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9476550.html 一.提出问题 不知道你是否遇到过这样的情况,在ssm框架中开发we ...
- C#基础系列:开发自己的窗体设计器(PropertyGrid显示中文属性名)
既然是一个窗体设计器,那就应该能够设置控件的属性,设置属性最好的当然是PropertyGrid了,我们仅仅需要使用一个PropertyGrid.SelectedObject = Control就可以搞 ...
- Spring基础系列--AOP织入逻辑跟踪
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9619910.html 其实在之前的源码解读里面,关于织入的部分并没有说清楚,那些前置.后 ...
- Spring基础系列-AOP源码分析
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9560803.html 一.概述 Spring的两大特性:IOC和AOP. AOP是面向切 ...
- Spring基础系列-容器启动流程(1)
原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9870339.html 概述 我说的容器启动流程涉及两种情况,SSM开发模式和Spri ...
- SpringBoot起飞系列-Web开发(四)
一.前言 从今天你开始我们就开始进行我们的web开发,之前的一篇用SpringBoot起飞系列-使用idea搭建环境(二)已经说明了我们如何进行开发,当然这是搭建起步,接下来我们就开始进行详细的开发, ...
- Spring Boot的web开发
web开发的自动配置类:org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration 自动配置的ViewResolver 视图的 ...
- 跟我学Spring Boot(三)Spring Boot 的web开发
1.Web开发中至关重要的一部分,Web开发的核心内容主要包括内嵌Servlet容器和SpringMVC spring boot 提供了spring-boot-starter-web 为web开发提 ...
- Spring Boot的web开发&静态资源配置方式
Web开发的自动配置类:org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration 1.1. 自动配置的ViewResolve ...
随机推荐
- 斜率优化入门学习+总结 Apio2011特别行动队&Apio2014序列分割&HZOI2008玩具装箱&ZJOI2007仓库建设&小P的牧场&防御准备&Sdoi2016征途
斜率优化: 额...这是篇7个题的题解... 首先说说斜率优化是个啥,额... f[i]=min(f[j]+xxxx(i,j)) ; 1<=j<i (O(n^2)暴力)这样一个式子,首 ...
- 【Canal源码分析】parser工作过程
本文主要分析的部分是instance启动时,parser的一个启动和工作过程.主要关注的是AbstractEventParser的start()方法中的parseThread. 一.序列图 二.源码分 ...
- 实验吧 ---- 隐写术之so beautiful so white
因为好久没有写博客,所以今天本宝宝要弥补这个过错,一下子更新许多文章,希望各位小伙伴能够原谅,以后我会加倍努力的! 这一次主要都是实验吧里面的 关于隐写术方面的知识,后续我会上传一些解密工具,希望能够 ...
- python里面的xlrd模块详解(一)
那我就一下面积个问题对xlrd模块进行学习一下: 1.什么是xlrd模块? 2.为什么使用xlrd模块? 3.怎样使用xlrd模块? 1.什么是xlrd模块? python操作excel主要用到xlr ...
- 危险 AI 花名册
简评:臭不要脸 AI 名单,another side of AI. 这是一个可怕的 AI 清单,上面的各种商用 AI 项目都用于一些很恶劣的目的.请大家保持警惕. 区别对待类 · HireVue - ...
- 百度推出 MIP Baidu Path链接
在站长将站点 MIP 化时,需要关注 URL 的一共有三个:MIP URL, MIP-Cache URL 以及 MIP Baidu Path. 从 URL 说起 在互联网中,URL 定义页面的地址,每 ...
- STM32学习笔记(一):跑马灯
本实验所采用的开发板为正点原子的MiniSTM32f103rc开发板,主函数程序如下,注释如下:main.c #include "stm32f10x.h" void Delay(u ...
- 如何查找元素对应事件的js代码,检测定位js事件
比如一张图片当鼠标放到上面时,图片改变.想找到这个事件对应的js代码,假设另存为html之后,文件夹中有.js文件. 如果你会调试,可以用打开浏览器的调试功能,以chrome为例,按F12打开调试窗口 ...
- 怎么构建vue-cli项目
1.安装node.js(已安装可直接跳过,建议查看node版本,node -v): 2.npm包管理器,是集成在node中的,可跳过(npm -v): 3.由于npm的有些资源被墙,为了更快更稳定,所 ...
- 版本号对比 -- Python实现
相同位数版本号大小比较: def abc(str1, str2): if str1 == "" or str2 == "": print("输入包含空 ...