SpringMVC学习笔记——1

SpringMVC学习笔记——1

  • 一、SpringMVC简介
    • 1.1、SpringMVC概述
    • 1.2、SpringMVC快速入门
    • 1.3、Controller中访问容器中的Bean
    • 1.4、SpringMVC关键组件的浅析
  • 二、SpringMVC的请求处理
    • 2.1、请求映射路径配置
    • 2.2、请求数据的接收
      • 2.2.1、键值对方式接收数据
      • 2.2.2、封装JavaBean数据
      • 2.2.3、接收JSON格式数据
      • 2.2.4、接收Restful风格数据
      • 2.2.5、接收上传的文件
      • 2.2.6、获取header头信息和cookie信息
      • 2.2.7、直接获取Requset和Session域中的数据
    • 2.3、Javaweb常用对象获取
    • 2.4、请求静态资源
    • 2.5、注解驱动\<mvc:annotation-driven>标签
  • 三、SpringMVC的响应处理
    • 3.1、传统同步业务数据响应
    • 3.2、前后端分离异步业务数据响应

一、SpringMVC简介

1.1、SpringMVC概述

SpringMVC是一个基于Spring开发的MVC轻量级框架,Spring3.0后发布的组件,SpringMVC和Spring可以无缝整合,使用DispatcherServlet作为前端控制器,且内部提供了处理器映射器、处理器适配器、视图解析器等组件,可以简化JavaBean封装,Json转化、文件上传等操作。
在这里插入图片描述

1.2、SpringMVC快速入门

在这里插入图片描述

  1. 导入spring-mvc坐标
<dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.3.7</version>
</dependency>
  1. 配置前端控制器DispatcherServlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"version="4.0"><!--配置DispatcherServlet--><servlet><servlet-name>DispatcherServlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class></servlet><servlet-mapping><servlet-name>DispatcherServlet</servlet-name><url-pattern>/</url-pattern></servlet-mapping>
</web-app>
  1. 编写Controller,配置映射路径,并交给SpringMVC容器管理
@Controller
public class QuickController {@RequestMapping("/show")private void show(){System.out.println("show running...");}
}

1.3、Controller中访问容器中的Bean

  1. 创建Bean
@Service
public class QuickServiceImpl implements QuickService {
}
  1. 用spring-xml文件扫描包含该Bean的包
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd
"><!--组件扫描--><context:component-scan base-package="com.Smulll.Service"/>
</beans>
  1. 在Web-xml文件中配置ContextLoaderListener
<!--配置ContextLoaderlistener-->
<context-param><param-name>contextConfigLocation</param-name><param-value>classpath:applicationContext.xml</param-value>
</context-param>
<listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
  1. 在Controller中访问容器中的Bean
@Controller
public class QuickController {/*直接注入Service进行使用*/@Autowiredprivate QuickService quickService;
}

1.4、SpringMVC关键组件的浅析

上面已经完成的快速入门的操作,也在不知不觉中完成的Spring和SpringMVC的整合,我们只需要按照规则去定义Controller和业务方法就可以。但是在这个过程中,肯定是很多核心功能类参与到其中,这些核心功能类,一般称为组件。当请求到达服务器时,是哪个组件接收的请求,是哪个组件帮我们找到的Controller,是哪个组件帮我们调用的方法,又是哪个组件最终解析的视图?

组件描述常用组件
处理器映射器:HandlerMapping匹配映射路径对应的Handler,返回可执行的处理器链对象HandlerExecutionChain对象RequestMappingHandlerMapping
处理器适配器:HandlerAdapter匹配HandlerExecutionChain对应的适配器进行处理器调用,返回视图模型对象RequestMappingHandlerAdapter
视图解析器:ViewResolver对视图模型对象进行解析InternalResourceViewResolver

在这里插入图片描述

二、SpringMVC的请求处理

2.1、请求映射路径配置

配置映射路径,映射器处理器才能找到Controller的方法资源,目前主流映射路径配置方式就是@RequestMapping

相关注解作用使用位置
@RequestMapping设置控制器方法的访问资源路径,可以接收任何请求方法和类上
@GetMapping设置控制器方法的访问资源路径,可以接收GET请求方法和类上
@PostMapping设置控制器方法的访问资源路径,可以接收POST请求方法和类上
  • 当给类加了@RequestMapping,想要访问某个方法就必须要在虚拟地址上加入类的RequestMapping名以及方法的RequestMapping名
@Controller
@RequestMapping("/quick")
public class QuickController {/*直接注入Service进行使用*/@Autowiredprivate QuickService quickService;//使用GET方法获取数据//@RequestMapping(value = {"/show","/showaaa"},method = RequestMethod.GET)@GetMapping("/show")//当给类加了@RequestMapping("/quick"),想访问show方法就必须要 虚拟地址+quick/showprivate String show(){System.out.println("show running..."+quickService);return "/index.jsp";}
}

2.2、请求数据的接收

2.2.1、键值对方式接收数据

接收普通请求数据,当客户端提交的数据是普通键值对形式时,直接使用同名形参接收即可

@Controller
public class ParamController {	//http://localhost:8080/SpringMVCDome1/Param4?username=zhangsan&age=12&hobby=football@RequestMapping("/Param4")public String Param4(@RequestParam Map<String,String> map){map.forEach((k,v)->{System.out.println(k+v);});return "index.jsp";}//http://localhost:8080/SpringMVCDome1/Param4?hobby=pingpang&hobby=football&hobby=basketball@RequestMapping("/Param3")public String Param3(@RequestParam List<String> hobby){for (String s : hobby) {System.out.println(s);}return "index.jsp";}//http://localhost:8080/SpringMVCDome1/Param4?hobby=pingpang&hobby=football&hobby=basketball@RequestMapping("/Param2")public String Param2(String[] hobby){for (String s : hobby) {System.out.println(s);}return "index.jsp";}//http://localhost:8080/SpringMVCDome1/Param4?username=zhangsan&age=23@RequestMapping("/Param1")public String Param1(@RequestParam("username") String name, int age){System.out.println(name+age);return "index.jsp";}
}
  • @RequestParam可以使获取的数据不创建对象而是直接注入形参,因此使用该注解可以使得参数不一样而注入
    • @RequestParam有三个属性
    • value:要注入参数的名字
    • required:该数据是否必须注入
      • true:是
      • false:否(默认)
    • defaultValue:当数据未提交时的默认值

2.2.2、封装JavaBean数据

接收实体JavaBean属性数据,单个JavaBean数据:提交的参数名称只要与Java的属性名一致,就可以进行自动封装

username=haohao&age=35&hobbies=eat&hobbies=sleep
	public class User{private string username ;private Integer age;private String [] hobbies;private Date birthday;private Address address;// ...省略get和set方法
}
@GetMapping("/show")
pubiic string show(User user){system.out.println(user);return "/index.jsp" ;
}

2.2.3、接收JSON格式数据

  • 使用POST请求方式,添加@RequestBody可以使json格式字符串整体打印出来,不会被解析
@PostMapping("/Param5")
public String Param5(@RequestBody String body) {System.out.println(body);return "index.jsp";
}

使用Json工具 ( jackson )将Json格式的字符串转化为JavaBean进行操作

<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.9.0</version>
</dependency>
@PostMapping ("/show" )
public string show(@RequestBody string body) throws IOException {system.out.println(body);//获得objectMapperObjectMapper objectMapper = new ObjectMapper();//将json格式字符串转化成指定的UserUser user = objectMapper.readValue(body,User.class);System.out.println(user);return "/index.jsp";
}
  • 配置HandlerAdapter可以自动执行转换的工作
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"><property name="messageConverters"><list><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"></bean></list></property>
</bean>
@PostMapping ("/show" )
public string show(@RequestBody User user) throws IOException {System.out.println(user);return "/index.jsp";
}

2.2.4、接收Restful风格数据

什么是Rest风格?
Rest (Representational State Transfer)表象化状态转变(表述性状态转变),在2000年被提出,基于HTTP、URI
、xml、JSON等标准和协议,支持轻量级、跨平台、跨语言的架构设计。是Web服务的一种新网络应用程序的设计风格和开发方式。

Restful风格的请求,常见的规则有如下三点:
用URI表示某个模块资源,资源名称为名词;

模板URI资源
用户模块userhttp://localhost/user
商品模块producthttp://localhost/product
账户模块accounthttp://localhost/account
日志模块loghttp://localhost/log

用请求方式表示模块具体业务动作,例如:GET表示查询、POST表示插入、PUT表示更新、DELETE表示删除

URI资源请求方式参数解释
http://localhost/user/100GET存在URL地址中:100查询id=100的User数据
http://localhost/userPOST存在请求体中Json: {“username”:“haohao” ,“age”:18}插入user数据
http://localhost/userPUT存在请求体中Json : {“id”:100,“username”:“haohao” , “age”:18}修改id=100的User数据
http://localhost/user/100DELETE存在URL地址中:100删除id=100的User数据
http://localhost/product/5GET存在URL地址中:5查询id=5的Product数据
http://localhost/productPOST存在请求体中Json: {“proName”:“小米手机” ,“price”:1299}插入Product数据
http://localhost/productPUT存在请求体中Json : {“id”:5,“proName”:“小米手机” ,“price”:1299}修改id=5的Product数据
http://localhost/product/5DELETE存在URL地址中:5删除id=5的Product数据
  • 用HTTP响应状态码表示结果,国内常用的响应包括三部分:状态码、状态信息、响应数据
{"code":200,"message":"成功","data":{"username":"haohao","age":18}
}
{"code":300,"message":"执行错误","data":"",
}

接收Restful风格数据,Restful请求数据一般会在URL地址上携带,可以使用注解@PathVariable(占位符参数
名称)

http://localhost/user/100
@PostMapping ("/user/{id}")
public string findUserById(@Pathvariable("id") Integer id){system.out.println(id);return "/index.jsp";
}

请求URL资源地址包含多个参数情况

http://localhost/user/haohao/18
@PostMapping ("/user/{username}/{age}")
public string findUserById(@Pathvariable("username") String username,@Pathvariable("age") Integer age){system.out.println(username+"=="+age);return "/index.jsp";
}

2.2.5、接收上传的文件

接收文件上传的数据,文件上传的表单需要一定的要求,如下:

  • 表单的提交方式必须是POST
  • 表单的enctype属性必须是multipart/form-data
  • 文件上传项需要有name属性
<form action="" enctype="multipart/form-data" method="post"><input type="file" name="myFile">
</form>
http://localhost:8080/SpringMVCDome1/Param6
@PostMapping("/Param6")
public String Param6(@RequestBody MultipartFile myFile) {System.out.println(myFile);return "/index.jsp";
}

服务器端,由于映射器适配器需要文件上传解析器,而该解析器默认未被注册,所以手动注册

<!--配置文件上传解析器,注意:id的名字是固定写法-->
<bean id="multipartResolver"
class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><property name="defaultEncoding" value="UTF-8"/><!--文件的编码格式默认是Iso8859-1--><property name="maxUploadSizePerFile" value="1048576"/><!--上传的每个文件限制的大小单位字节--><property name="maxUploadSize" value="3145728"/><!--上传文件的总大小--><property name="maxInMemorySize" value="1048576"/><!--上传文件的缓存大小-->
</bean>

而CommonsMultipartResolver底层使用的Apache的是Common-fileuplad等工具API进行的文件上传

<dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.4</version>
</dependency>
@PostMapping("/Param6")
public String Param6(@RequestBody MultipartFile myFile) throws IOException {System.out.println(myFile);//将上传的文件进行保存//1、获取当前上传的文件的输入流InputStream inputStream = myFile.getInputStream();//2、获得上传文件位置的输出流OutputStream outputStream = new FileOutputStream("D:\\"+"aaa");//3、执行文件拷贝IOUtils.copy(inputStream,outputStream);outputStream.close();inputStream.close();return "/index.jsp";
}

2.2.6、获取header头信息和cookie信息

接收Http请求头数据,接收指定名称的请求头

GetMapping ("/headers")
public string headers(@RequestHeader("Accept-Encoding") String acceptEncoding){System.out.println ( "Accept-Encoding:"+acceptEncoding);return "/index.jsp";

接收所有的请求头信息

@GetMapping ( "/ headersMap" )
public string headersMap(@RequestHeader Map<String,string> map){map.forEach((k, v)->{system.out.println(k+":"+V);});return " /index.jsp" ;

获得客户端携带的Cookie数据

GetMapping("/cookies")
public String cookies(@CookieValue(value = "JSESSIONID",defaultValue = "") String jsessionid){system.out.println(jsessionid);return "/index.jsp" ;
}

2.2.7、直接获取Requset和Session域中的数据

获得转发Request域中数据,在进行资源之间转发时,有时需要将一些参数存储到request域中携带给下一个资源

GetMapping ("/request1")
public string request1(HttpServletRequest request) {//存储数据request.setAttribute("username","haohao");return "/request2";
}
GetMapping ("/request2")
public string request2(@RequestAttribute("username") String username) {system.out.println(username);return "/index.jsp";
}

2.3、Javaweb常用对象获取

直接打印参数request和response,可以发现这两个参数是SpringMVC自动传入的

@GetMapping("/Param10")
public String Param10(HttpServletRequest request, HttpServletResponse response) {System.out.println(request);System.out.println(response);return "/index.jsp";
}

在这里插入图片描述

2.4、请求静态资源

在Spring-MVC框架下,无法直接访问到web文件下的静态资源

静态资源请求的三种解决方案:

  • 第一种方案,可以再次激活Tomcat的DefaultServlet,Servlet的url-pattern的匹配优先级是:精确匹配>目录匹配>扩展名匹配>缺省匹配,所以可以指定某个目录下或某个扩展名的资源使用DefaultServlet进行解析:
<servlet-mapping><servlet-name>default</servlet-name><url-pattern>*.html</url-pattern>
</servlet-mapping>
<servlet-mapping><servlet-name>default</servlet-name><url-pattern>/img/*</url-pattern>
</servlet-mapping>
  • 第二种方式,在spring-mvc.xml中去配置静态资源映射,匹配映射路径的请求到指定的位置去匹配资源
<!--mapping是映射资源路径,location是对应资源所在的位置-->
<mvc:resources mapping="/img/*" location="/img/" />
<mvc:resources mapping="/css/*" location="/css/" />
<mvc:resources mapping="/js/*" location="/js/" />
<mvc:resources mapping="/html/*" location="/html/" />
  • 第三种方式,在spring-mvc.xml中去配置<mvc:default-servlet-handler >,该方式是注册了一个
    DefaultServletHttpRequestHandler处理器,静态资源的访问都由该处理器去处理,这也是开发中使用最多的
<mvc:default-servlet-handler/>

2.5、注解驱动<mvc:annotation-driven>标签

该标签内部会帮我们注册RequestMappingHandlerMapping、注册RequestMappingHandlerAdapter并注入Json消息转换器等,上述配置就可以简化成如下:

<!--mvc注解驱动-->
<mvc:annotation-driven/>
<!--配置DefaultServletHttpRequestHandler-->
<mvc:default-servlet-handler/>

PS: <mvcannotation-driven>标签在不同的版本中,帮我们注册的组件不同,Spring 3.0.X版本注册是
DefaultAnnotationHandlerMappingAnnotationMethodHandlerAdapter,由于框架的发展,从Spring 3.1.X开始注册组件变为RequestMappingHandlerMappingRequestMappingHandlerAdapter.

三、SpringMVC的响应处理

3.1、传统同步业务数据响应

Spring的接收请求的部分我们讲完了,下面在看一下Spring怎么给客户端响应数据,响应数据主要分为两大部分:

  • 传统同步方式:准备好模型数据,在跳转到执行页面进行展示,此方式使用越来越少了,基于历史原因,一些旧项目还在使用;
  • 前后端分离异步方式:前端使用Ajax技术+Restful风格与服务端进行Json格式为主的数据交互,目前市场上几乎都是此种方式了。

传统同步业务数据响应

  • 请求资源转发
  • 请求资源重定向
  • 响应模型数据
  • 直接回写数据给客户端

在这里插入图片描述

  • 转发的关键字forword
  • 重定向的关键字redirect

通过视图模型的方式进行转发数据

@Controller
public class QuickController1 {@GetMapping("/res1")//当给类加了@RequestMapping("/quick"),想访问show方法就必须要 虚拟地址+quick/showprivate ModelAndView show(ModelAndView modelAndView){User user = new User();user.setUsername("zhangsan");user.setAge(18);user.setGender('男');modelAndView.addObject("user",user);modelAndView.setViewName("/index.jsp");return modelAndView;}
}

直接回写字符串

需要使用Tomcat8

@RequestMapping("/res2")
@ResponseBody//告诉SpringMVC返回的字符串不是视图名,是以响应体方式响应的数据
public String res2(){return "hello,world";
}

3.2、前后端分离异步业务数据响应

其实此处的回写数据,跟上面回写数据给客户端的语法方式一样,只不过有如下一些区别:

  • 同步方式回写数据,是将数据响应给浏览器进行页面展示的,而异步方式回写数据一般是回写给Ajax引擎的,即谁访问服务器端,服务器端就将数据响应给谁
  • 同步方式回写的数据,一般就是一些无特定格式的字符串,而异步方式回写的数据大多是Json格式字符串

回写普通数据使用@ResponseBody标注方法,直接返回字符串即可,此处不在说明;
回写Json格式的字符串,即将直接拼接Json格式的字符串或使用工具将JavaBean转换成Json格式的字符串回写

@GetMapping("/response3")
@ResponseBody
public string response3(HttpServletResponse response){return "{\"username\":\"haohaol\",\"age\":18}";
}
@GetMapping("/response4")
@ResponseBody
public string response4() throws JsonProcessingException {//创建JavaBeanUser user = new User();user.setUsername("haohao");user.setAge(18);//使用Jackson转换成json格式的字符串String json = new objectMapper().writevalueAsString(user);return json;
}

使用@RestController 相当于@ResponseBody和@Controller结合

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/115828.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue之 h() 函数

前言 Vue推荐在绝大数情况下使用模板来创建HTML&#xff0c;然后一些特殊的场景&#xff0c;你真的需要JavaScript的完全编程的能力&#xff0c;这个时候你可以使用渲染函数 &#xff0c;它比模板更接近编译器&#xff1b; h()函数是什么 Vue在生成真实的DOM之前&#xff0c…

时间轮算法

思考 假如现在有个任务需要3s后执行&#xff0c;你会如何实现&#xff1f; 线程实现&#xff1a;让线程休眠3s 如果存在大量任务时&#xff0c;每个任务都需要一个单独的线程&#xff0c;那这个方案的消耗是极其巨大的&#xff0c;那么如何实现高效的调度呢&#xff1f; 时…

华为云云耀云服务器 L 实例评测|配置教程 + 用 Python 简单绘图

文章目录 Part.I IntroductionChap.I 云耀云服务器 L 实例简介Chap.II 参与活动步骤 Part.II 配置Chap.I 初步配置Chap.II 配置安全组 Part.III 简单使用Chap.I VScode 远程连接华为云Chap.II 简单绘图 Reference Part.I Introduction 本篇博文是为了参与华为“【有奖征文】华…

Linux常用命令—find命令大全

文章目录 一、find命令常用功能1、find命令的基本信息如下。2、按照文件名搜索3、按照文件大小搜索4、按照修改时间搜索5、按照权限搜索举例&#xff1a;6、按照所有者和所属组搜索7、按照文件类型搜索8、逻辑运算符 一、find命令常用功能 1、find命令的基本信息如下。 命令名…

招商信诺人寿基于 Apache Doris 统一 OLAP 技术栈实践

本文导读&#xff1a; 当前&#xff0c;大数据、人工智能、云计算等技术应用正在推动保险科技发展&#xff0c;加速保险行业数字化进程。在这一背景下&#xff0c;招商信诺不断探索如何将多元数据融合扩充&#xff0c;以赋能代理人掌握更加详实的用户线索&#xff0c;并将智能…

七分钟,数据转换器get到了

全文阅读时间 | 预计七分钟 KING BASE 开源 OR 闭源&#xff1f; 在瞬息多变的软件市场上&#xff0c;开源还是闭源是一个恒久不变的话题。开源软件得益于基础架构和基本功能的全面开放&#xff0c;开发者能自由使用和二次开发&#xff0c;但使用前提是需要投入大量成本对软件进…

基于Vue+ELement搭建登陆注册页面实现后端交互

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是Java方文山&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的专栏《ELement》。&#x1f3af;&#x1f3af; &#x1…

【观察】华为陈帮华:学习“都江堰模式”,构筑强健分销体系

目前在水利生态构建上&#xff0c;最有特点无疑是都江堰工程&#xff0c;它是以无坝引水为特征的水利生态工程&#xff0c;从无坝引水到平原灌溉&#xff0c;均有口无闸&#xff0c;既充分利用了河道的弯曲和崖壁的角度&#xff0c;自然控制流向和水量&#xff0c;同时在不改变…

【数据结构】—交换排序之快速排序究极详解,手把手带你从简单的冒泡排序升级到排序的难点{快速排序}(含C语言实现)

食用指南&#xff1a;本文在有C基础的情况下食用更佳 &#x1f525;这就不得不推荐此专栏了&#xff1a;C语言 ♈️今日夜电波&#xff1a;靴の花火—ヨルシカ 0:28━━━━━━️&#x1f49f;──────── 5:03 …

自动化测试的定位及一些思考

大家对自动化的理解&#xff0c;首先是想到Web UI自动化&#xff0c;这就为什么我一说自动化&#xff0c;公司一般就会有很多人反对&#xff0c;因为自动化的成本实在太高了&#xff0c;其实自动化是分为三个层面的&#xff08;UI层自动化、接口自动化、单元测试&#xff09;&a…

电巢科技出席第26届西北地区电子技术与线路课程教学改革研讨会,聚焦一流课程建设!

2023年9月15日至17日&#xff0c;北方民族大学召开第26届西北地区电子技术与线路课程教学改革研讨会。本次会议围绕“梳理课程教学内容&#xff0c;改革教学方式&#xff0c;探索虚拟教研室构建方式&#xff0c;完善基层教学组织&#xff0c;推进一流课程和一流教材资源共享&am…

Java 基于微信小程序的学生选课系统

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝30W、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 文章目录 第一章&#xff1a; 简介第二章 技术栈第三章&#xff1a; 功能分析第四章 系统设计第五章 系统功…