Hello,明天给各位童鞋们分享 Spring MVC,连忙拿出小本子记下来吧!
1.SpringMVC 概述
1.1 三层架构
三层架构:
体现层:负责数据展现
业务层:负责业务解决
数据层:负责数据操作
1.2 MVC
MVC(Model View Controller),一种用于设计创立 web 应用程序体现层的模式
Model(模型):数据模型,用户封装数据
View(视图):页面视图,用户展现数据
- jsp
- html
- Controller(控制器):解决用户交互的调度器,用于依据用户需要处理程序逻辑
- Servlet
- SpringMVC
2. 入门案例
2.1 入门案例制作(重点)
XML 版
- XML+ 注解版(主体)
- 纯注解版(变形)
- 基于 Spring 环境开发
步骤:
导入坐标
<dependencies>
<!–servlet3.1 标准坐标 –>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
<scope>provided</scope>
</dependency>
<!–jsp 坐标 –>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.1</version>
<scope>provided</scope>
</dependency>
<!–spring 坐标 –>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.1.9.RELEASE</version>
<scope>provided</scope>
</dependency>
<!–springmvc 坐标 –>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.1.9.RELEASE</version>
<scope>provided</scope>
</dependency>
<!–spring web 坐标 –>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.1.9.RELEASE</version>
<scope>provided</scope>
</dependency>
</dependencies>
<!– 构建 –>
<build>
<!– 设置插件 –>
<plugins>
<!– 具体的插件配置 –>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.1</version>
<configuration>
<port>80</port>
<path>/</path>
</configuration>
</plugin>
</plugins>
</build>
2. 定义业务层处理器 Controller,并配置成 spring 的 bean(等同于 Servlet)
该 bean 的解决须要应用独立的配置文件扫描(XML 版):spring-mvc.xml
3.web.xml 中配置 SpringMVC 外围控制器,用于将申请转发到对应的具体业务处理器 Controller 中(等同于 Servlet 配置)
4. 设定具体 Controller 的拜访路劲(等同于 Servlet 在 web.xml 中的配置),并设置返回页面
此处记录一个问题:
问题景象:org.apache.catalina.core.ContainerBase.addChildInternal ContainerBase.addChild: start: org.apache.catalina…
问题解决:
查看 web.xml 是否无误,门路等是否有误,一个 servlet 不能同时应用 xml 和注解配置,否则也很会呈现该谬误;
查看我的项目 lib 目录是否存在,idea 默认不创立,
进入 Project Structures(ctrl+shift+alt+S)–> 点击左侧 Project Settings 下的 Artifacts。
点击两头栏我的项目,这里会两个文件,一个是:我的项目名:war(war 压缩包),一个是我的项目名:war exploded(war 未压缩包)。
点击 war exploded 我的项目,在右侧中第一栏 Output Layout(我的项目公布生成的文件)下,开展 WEB-INF 文件夹,此时该目录下只有 classes 目录,无 lib 目录,本人手动创立一个 lib 目录,并点击下面 + 抉择 Library File 增加 maven 导入的 jar 包
重启 tomcat 就解决了(我是这样解决的,不保障所有相似问题都能解决)
2.2 入门案例工作流程剖析(重点)
服务器启动:
加载 web.xml 中的 DispatcherServlet
读取 spring-mvc.xml 中的配置,加载所有 com.itheima 包中所有标记为 bean 的类
读取 bean 中办法上标注 @RequestMapping(“/save”)的内容
解决申请:
DispatcherServlet 配置拦挡所有申请 /
应用申请门路与所有加载的 @RequestMapping 的内容进行比对
执行对应的办法
依据办法的返回值在 webapp 目录中查找对应的页面并展现
2.3 SpringMVC 技术架构图(重点)
SpringMVC 技术架构图:
DispatcherServlet:前端控制器,是整体流程控制中心,由其调用其余组件解决用户的申请,无效的升高了组件间的耦合性
HandleMapping:处理器映射器,负责依据用户申请找到对应具体的 Handler 处理器
Handler:处理器,业务解决的外围类,通常由开发者编写,形容具体的业务
HandlerAdapter:处理器适配器,通过它对处理器进行执行
ViewResolver:视图解析器,将处理结果生成 View 视图
View:视图,最终产出后果,罕用视图入 jsp、html
3. 根底配置
3.1 Controller 加载管制(重点)
SpringMVC 的处理器对应 bean 必须依照标准格局开发,为防止退出有效的 bean 可通过 bean 加载过滤器进行蕴含设定或排除设定,体现层 bean 标注通常设为 @Controller
<context:component-scan base-package=”com.itheima”>
<context:include-filter
type=”annotation”
expression=”org.springframework.stereotype.Controller”/>
</context:component-scan>
此处记录一个问题:
问题景象:启动 tomcat 实现时,idea 弹窗提醒“http://localhost:80/ 找不到应 …
解决办法:点击 idea 主界面运行图标旁下拉列表,抉择 Edit Configuration,进入配置界面,Open Browser 抉择一个固定浏览器,不要应用默认的浏览器(tomcat 启动实现主动应用该浏览器拜访)
制作案例:和后面案例一样,只需批改 spring-mvc.xml 配置文件:
<context:component-scan base-package=”com.itheima”>
<!– 增加过滤,蕴含该注解才会被扫描到 –>
<context:include-filter type=”annotation” expression=”org.springframework.stereotype.Controller”/>
</context:component-scan>
bean 加载管制阐明:
业务层与数据层 bean 加载由 spring 管制,参照 spring 课程加载形式
体现层 bean 加载由 SpringMVC 独自管制
体现层处理器 bean 应用注解 @Controller 申明
bean 加载管制应用蕴含性过滤器
过滤器类型为通过注解进行过滤
过滤的注解名称为 Controller
3.2 动态资源加载
外围控制器拦挡的是所有申请,须要对非一般资源申请进行放行,通过配置放行资源实现
<mvc:resources mapping=”/img/**” location=”/img/”/>
<mvc:resources mapping=”/js/**” location=”/img/”/>
<mvc:resources mapping=”/css/**” location=”/img/”/>
应用简化格局能够放行所有一般资源调用,无需一一枚举
<mvc:default-servlet-handler/>
还是之前案例内容,只需做如下批改:
在 webap 下新建 img 目录,并导入一张图片
批改 spring-mvc.xml 配置文件:
留神:webapp 下 img 目录需设置为 resources 目录,否则无法访问
3.3 中文乱码解决
SpringMVC 提供专用的中文字符过滤器,用于解决乱码问题。在 web.xml 进行如下配置:
3.4 注解驱动
应用注解模式转化 SpringMVC 外围配置文件为配置类
@Configuration
@ComponentScan(value = “com.itheima”,
excludeFilters = @ComponentScan.Filter(
type = FilterType.ANNOTATION,
classes = Controller.class)
)
public class SpringMvcConfiguration {
}
操作过程:
创立配置类
- 基于 servlet3.0 标准,自定义 servlet 容器初始化配置类,加载 SpringMVC 外围配置类
- 动态资源加载过滤(注解版)
配置类实现 WebMvcConfigurer 接口,笼罩 addResourceHandlers 办法,在其中对具体的资源进行设定
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler(“/img/**”).addResourceLocations(“/img/”);
registry.addResourceHandler(“/js/**”).addResourceLocations(“/js/”);
registry.addResourceHandler(“/css/**”).addResourceLocations(“/css/”);
}
或者笼罩 configureDefaultServletHandling 办法,应用 Servlet 默认过滤规定
@Override
public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
configurer.enable();
}
4. 中文乱码解决(注解版)
Servlet3.0 标准启动服务器时做的工作通过实现 ServletContainerInitializer 接口,在 onStartup 办法中实现,包含监听器注册、过滤器注册等
4. 申请
4.1 申请参数(重点)
SpringMVC 将传递的参数封装到处理器办法的形参中,达到快速访问参数的目标
拜访 URL:http://localhost/requestParam…
@RequestMapping(“/requestParam”)
public String requestParam(String name) {
System.out.println(“name = “+name);
return “page.jsp”;
}
申请参数类型:
一般类型参数
POJO 类型参数
数组类型参数
汇合类型参数
一般类型参数
参数名与处理器办法形参名保持一致
拜访 URL:http://localhost/requestParam…
参数设定
名称:@RequestParam
类型:形参注解
地位:处理器类中的办法形参后方
作用:绑定申请参数与对应解决办法形参间的关系
范例:
@RequestMapping(“/requestParam2”)
public String requestParam2(@RequestParam(value = “userName”,required = true,defaultValue = “value”) String name) {
System.out.println(“name = “+name);
return “page.jsp”;
}
留神:若设置了 required = true,则在 浏览器 URL 地址栏必须加上参数,否则报错 404。能够通过设置默认参数 defaultValue = “value” 防止未输出报错
拜访门路:http://localhost/requestParam…
POJO 类型参数
当 POJO 中应用简略类型属性时,参数名称与 POJO 类属性名保持一致
拜访 URL:http://localhost/requestParam…
留神:这里须要正文掉下面的办法 requestParam2,否者此处也须要给参数增加注解 @RequestParam,否则报错 500:java.lang.IllegalArgumentException: Name for argument type [java.lang.String] not available, and par…
参数抵触
当 POJO 类型属性与其余参数呈现同名问题时,将被同时笼罩
拜访 URL:http://localhost/requestParam…
控制台打印后果:user = User{name=‘Jock’, age=30}, name = Jock
倡议应用 @RequestParam 注解进行辨别
留神:这里也须要正文上一个办法,否者报错 500:java.lang.IllegalStateException: Method [requestParam4] was discovered in the .class file but cannot be resolved in the class object
简单 POJO 类型参数
当 POJO 中呈现对象属性时,参数名称与对象层次结构名称保持一致
拜访 URL:http://localhost/requestParam…
public class Address {
private String province;
private String city;
private String address;
}
public class User {
private String name;
private Integer age;
private Address address;
}
@RequestMapping(“/requestParam5”)
public String requestParam5(User user) {
System.out.println(“city = “+user.getAddress().getCity());
return “page.jsp”;
}
控制台打印后果:city = bejing
当 POJO 中呈现汇合,保留简略数据,应用多个雷同名称的参数为其进行赋值
URL:http://localhost/requestParam…
控制台打印后果:user = User{name=‘null’, age=null, address=null, nick=[Jock1, Jock2, Jocker]}
当 POJO 中呈现 List,保留对象数据,参数名称与对象层次结构名称保持一致,应用数组格局形容汇合中对象的地位
URL:http://localhost/requestParam…
@RequestMapping(“/requestParam7”)
public String requestParam7(User user) {
System.out.println(“addresses = “+user.getAddresses());
return “page.jsp”;
}
拜访报错 400:java.lang.IllegalArgumentException: 在申请指标中找到有效字符。无效字符在 RFC 7230 和 RFC 3986 中定义
百度解决方案:tomcat 配置文件–>conf–>server.xml,批改内容
<Connector port=”8080″ protocol=”HTTP/1.1″ connectionTimeout=”20000″ redirectPort=”8443″ URIEncoding=”utf-8″
relaxedPathChars=”|{}[],%”
relaxedQueryChars=”|{}[],%” />
然而,批改了还是报该谬误
当 POJO 中呈现 Map,保留对象数据,参数名称与对象层次结构名称保持一致,应用映射格局形容汇合对象的地位
URL:http://localhost/requestParam…
同样报错
数组类型参数
申请参数名与处理器办法形参名保持一致,且申请参数数量 >1 个
拜访 URL:http://localhost/requestParam…
控制台打印后果:name = jock name = jocker
汇合类型参数
保留简略类型数据,申请 参数名与处理器办法形参名保持一致,且申请参数数量 >1 个
拜访 URL:http://localhost/requestParam…
控制台打印后果:nick = [jock, jocker]
留神:SpringMVC 默认将 List 作为对象解决,赋值前先创建对象,而后将 nick 作为对象的属性进行解决。因为 List 是接口,无奈创建对象,报错无奈找到构造方法异样;修复类型可创建对象的 ArrayList 类型后,对象能够创立,但没有 nick 属性,因而数据为空。此时须要告知 SpringMVC 的处理器 nick 是一组数据,而不是一个繁多数据。通过 @RequestParam 注解,将数量大于 1 个 names 参数打包成参数数组后,SpringMVC 能力辨认该数据格式,并断定形参类型是否为数组或汇合,并依照数组或汇合对象的模式操作数据。
4.2 类型转换器
SpringMVC 对接管的数据进行主动类型转换,该工作通过 Converter 接口实现
标量转换器:
StringToBooleanConverter String → Boolean
ObjectToStringConverter Object → String
StringToNumberConverterFactory String → Number(Integer、Long 等)
NumberToNumberConverterFactory Number 子类型之间 (Integer、Long、Double 等)
StringToCharsetConverter String → java.lang.Character
NumberToCharacterConverter Number 子类型(Integer、Long、Double 等) → java.lang.Character
CharacterToNumberFactory java.lang.Character → Number 子类型(Integer、Long、Double 等)
StringToEnumConverterFactory String → Enum 类型
EnumToStringConverter Enum 类型 → String
StringToLocaleConverter String → java.util.Local
PropertiesToStringConverter java.util.Properties → String
StringToPropertiesConverter String → java.util.Properties
汇合、数组相干转换器
ArrayToCollectionConverter 数组 → 汇合(List、Set)
…
默认转换器
ObjectToObjectConverter Object 间
IdToEntityConverter Id → Entity
FallbackObjectToStringConverter Object → String
拜访 URL:http://localhost/requestParam…
控制台打印后果:date = Fri Apr 30 00:00:00 CST 2021
当输出格局变换会报错,如:http://localhost/requestParam…
日期类型格局转换
批改 spring-mvc.xml 配置使输出格局满足需要
日期类型格局转换(简化版)
名称:DateTimeFormat
类型:形参注解、成员变量注解
地位:形参后面 或 成员变量上方
作用:为以后参数或变量指定类型转换规定
范例:
@RequestMapping(“/requestParam12”)
public String requestParam12(@DateTimeFormat(pattern = “yyyy-MM-dd”) Date date) {
System.out.println(“date = “+date);
return “page.jsp”;
}
@DateTimeFormat(pattern = “yyyy-MM-dd”)
private Date birthday;
留神:依赖注解驱动反对
<mvc:annotation-driven />
自定义类型转换器
自定义类型转换器,实现 Converter 接口,并指定转换前与转换后的类型
注册自定义转换器
通过注册自定义转换器,将该性能退出到 SpringMVC 的转换服务 ConverterService 中
4.3 申请映射(重点)
名称:@RequestMapping
类型:办法注解
地位:处理器类中办法定义上方
作用:绑定申请地址与对应解决办法间的关系
范例:
@RequestMapping(“/requestURL”)
public String requestParam12(@DateTimeFormat(pattern = “yyyy-MM-dd”) Date date) {
System.out.println(“date = “+date);
return “page.jsp”;
}
拜访门路:/requestURL
申请返回的页面地址默认为以后门路
当设置了公共的拜访前缀后,以后门路产生了变动,须要依据变动批改地址或批改拜访页面的门路
拜访 root 门路下的页面
@Controller
public class UserController {
@RequestMapping(“/requestURL1”)
public String requestURL1() {
System.out.println();
return “page.jsp”;
}
}
拜访 root 门路下的 user 门路下的页面
@Controller
@RequestMapping(“/user”)
public class UserController {
@RequestMapping(“/requestURL2”)
public String requestURL2() {
System.out.println();
return “/page.jsp”;
}
}
@RequestMapping 属性
罕用属性
value
method
5. 响应
5.1 无数据跳转页面
响应形式:(数据流)
页面
HTML(页面)
JSP(页面 + 数据)
…
数据
JSON 数据
XML 数据
文本数据
文件
数据流
页面跳转设定:
当处理器办法的返回值类型为 String 类型,即可通过具体的返回字设置拜访的页面
页面跳转形式:
转发(默认)
@RequestMapping(“/showPage1”)
public String showPage1() {
System.out.println(“user mvc controller is running…”);
return “forward:page.jsp”;
}
重定向
@RequestMapping(“/showPage2”)
public String showPage2() {
System.out.println(“user mvc controller is running…”);
return “redirect:page.jsp”;
}
留神:页面拜访地址所携带的 / 能够应用。然而在 WEB-INF 目录下的资源 forward 能够拜访,而 redirect 不能拜访。
页面拜访快捷设定
展现页面的保留地位通常固定,且构造类似,能够设定通用的拜访门路,简化页面配置格局(spring-mvc.xml)
<bean class=”org.springframework.web.servlet.view.InternalResourceViewResolver”>
<property name=”prefix” value=”/WEB-INF/page/”/>
<property name=”suffix” value=”.jsp”/>
</bean>
@RequestMapping(“/showPage3”)
public String showPage3() {
System.out.println(“user mvc controller is running…”);
return “page”;
}
应用了简化配置,跳转页面名称前不能加 forward,否则报错
@RequestMapping(“/showPage4”)
public String showPage4() {
System.out.println(“user mvc controller is running…”);
return “forward:page”;//HTTP 状态 404 – 未找到
}
页面拜访快捷设定缺省页面:
如果设定了返回值,应用 void 类型,则默认应用拜访门路作页面地址的前缀后缀
@RequestMapping(“/showPage5”)
public void showPage5() {
}
等同于
@RequestMapping(“/showPage5”)
public String showPage5() {
return “showPage5”;
}
5.2 带数据跳转页面(重点)
形式一:应用 HttpServletRequest 类型形参进行数据传递
@RequestMapping(“showPageAndData1”)
public String showPageAndData1(HttpServletRequest request) {
request.setAttribute(“name”,”itheima”);
return “page”;
}
形式二:应用 Model 类型形参进行数据传递
@RequestMapping(“showPageAndData2”)
public String showPageAndData2(Model model) {
model.addAttribute(“name”,”Jock”);
Book book = new Book();
book.setName(“SpringMVC 入门案例 ”);
book.setPrice(66.66);
model.addAttribute(book);
return “page”;
}
形式三:应用 ModelAndView 类型形参进行数据传递,将该对象作为返回字传递给调用者
重定向:
@RequestMapping(“showPageAndData5”)
public ModelAndView showPageAndData5(ModelAndView modelAndView) {
modelAndView.setViewName(“redirect:page”);// 等同于 return “redirect:page.jsp”;
return modelAndView;
}
留神:应用重定向须要正文后面设置的 spring-mvc.xml 配置中增加的 WEB-INF/page 目录,因为 WEB-INF 有平安拜访,重定向不能拜访。
5.3 存数据返回(JSON)(重点)
导入 json 坐标 pom.xml
运行 tomcat 可能会报错:rg.apache.catalina.core.StandardContext.filterStart 启动过滤器异样
java.lang.ClassNotFoundException: org.springframework.web.filter.CharacterEncodingFilter
解决方案:在下面讲过,Project Structures(ctrl+shift+alt+S)–> Project Settings 下的 Artifact –> swar exploded 我的项目 –> Output Layout –> 开展 WEB-INF–> 手动创立一个 lib 目录(因为下面曾经创立了,此处不再创立),并点击下面 + 抉择 Library File 增加 maven 导入的 jar 包。重启 tomcat 就解决了
返回数据:
形式一:应用 response 对象实现数据返回
@RequestMapping(“/showData1”)
public void showData1(HttpServletResponse response) throws IOException {
response.getWriter().write(“message”);
}
形式二:(简化格局)
@RequestMapping(“/showData2”)
@ResponseBody
public String showData2() {
return “message”;
}
返回 JSON 数据:
形式一:基于 response 返回数据的简化格局,返回 json 数据
@RequestMapping(“/showData2”)
@ResponseBody
public String showData2() {
return “{‘name’:’Jock’}”;
}
形式二(对象):应用 SpringMVC 提供的音讯类型转换器将对象与汇合数据主动转换为 json 数据
spring-mvc.xml 增加注解驱动:<mvc:annotation-driven/>
形式三(汇合):应用 SpringMVC 注解驱动简化配置
<mvc:annotation-driven/>
注解驱动格局
@Configuration
@ComponentScan(“com.itheima”)
public class SpringMVCConfiguration {
}
6.Servlet 相干接口
6.1 HttpServletRequest
SpringMVC 提供拜访原始 Servlet 接口 API 的性能,通过形参申明即可
打印后果:
org.apache.catalina.connector.RequestFacade@1f787c3e
org.apache.catalina.connector.ResponseFacade@45b7d257
org.apache.catalina.session.StandardSessionFacade@411f594e
6.2 HttpServletResponse
6.3 HttpSession
6.4 Head
Head 数据获取:
名称:@RequestHeader
类型:形参注解
地位:处理器类中的办法形参后方
作用:绑定申请头数据与对应解决办法形参间的关系
范例:
打印后果:3AE4C46FC4AE80D62FAA29DA2DDD383F
6.5 Session
Session 数据获取:
名称:@SessionAttribute
类型:形参注解
地位:处理器类中的办法形参后方
作用:绑定申请 Session 数据与对应解决办法形参间的关系
范例:
打印后果:itheima
Session 数据设置(理解):
名称:@SessionAttributes
类型:类注解
地位:处理器类上方
作用:申明放入 Session 范畴的变量名称,实用于 Model 类型数据传参
范例:
6.6 注解式参数数据封装底层原理
数据的起源不同,对应的解决策略要进行辨别
Head
Cookie
…
SpringMVC 应用策略模式进行解决散发
顶层接口:HandlerMethodArgumentResolver
实现类:…
好啦,明天的文章就到这里,心愿能帮