SpringMVC获取请求参数
通过servletAPI获取(HttpServletRequest的方式)
通过控制器方法的形参获取请求参数
***在控制器方法的形参位置,设置和请求参数同名的形参,当浏览器发送请求,匹配到请求映射时,在 DispatcherServlet中就会将请求参数赋值给相应的形参*****
如果参数名与形参不一致,那么就可以通过参数注解@RequestParam (“参数”)对应形参的方法
@RequestParam注解的三个属性
@RequestParam是将请求参数和控制器方法的形参创建映射关系
@RequestParam注解一共有三个属性:
- value:指定为形参赋值的请求参数的参数名
- required:设置是否必须传输此请求参数,默认值为true
- defaultValue:不管required属性值为true或false,当value所指定的请求参数没有传输或传输的值 为””时,则使用默认值为形参赋值
@RequestHeader : 将请求头信息和控制器方法的形参绑定
注解一共有三个属性:value、required、defaultValue,用法同@RequestParam
逻辑简写
@RequestMapping(“ /XXX” ) pubic String getHeader( @RequestParam(value= “name”…)String name @RequestHeader(value = “referer” ) String referer ){ sout(“name = “ + name); sout(“referer = “+ referer); return “XXX”; //请求跳转的逻辑页面 }
@Cookievalue : 将cookie数据和控制请求的形参绑定
@CookieValue注解一共有三个属性:value、required、defaultValue,用法同@RequestParam
经常用的方法–通过实体类中的属性名与请求参数的名一致 来获取请求参数
/** * 保证User类中的属性名与请求参数中的参数保持一致 * @param user * @return */ @RequestMapping("/param/pojo") public String pojoRequest(User user){ System.out.println(user); return "success"; }
解决请求参数获取乱码的问题 :
此时不能通过request.setCharacterEnCoding(”UTF-8“)
- 可以将请求设置为get,因为get会自动设置成UTF-8模式
- 或者在web.xml中设置处理编码的过滤器【一定要匹配在其他配置之前】(四个)
<!--配置springMVC的编码过滤器--> <filter> <filter-name>CharacterEncodingFilter</filter-name> <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> <init-param> <param-name>encoding</param-name> <param-value>UTF-8</param-value> </init-param> <init-param> <param-name>forceEncoding</param-name> <param-value>true</param-value> </init-param> </filter> <filter-mapping> <filter-name>CharacterEncodingFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
往request域对象中共享数据
使用serletAPI向request域对象共享数据
使用ModelAndView向request域对象中共享数据★★★★
使用这个就必须将请求方法的返回值 设置为modelAndiew
其实不管是使用String也好,还是servletAPI也好,还是其他的 model也好,最后在底层都会被封装成ModelAndView 对象★★★★
@RequestMapping("/test/mav") public ModelAndView testModelAndView(){ /** * ModelAndView有Model和View的功能 * Model主要用于向请求域共享数据 * View主要用于设置视图,实现页面跳转 */ ModelAndView mav = new ModelAndView(); //向请求域共享数据 mav.addObject("testScope", "hello,ModelAndView"); //设置逻辑视图,实现页面跳转 mav.setViewName("success"); return mav; }
使用model向request域对象中共享数据
@RequestMapping("/test/model") public String testModel(Model model){ model.addAttribute("testModel","hello,model"); return "success"; }
使用map向request域对象中共享数据
使用modelmap向request域对象中共享数据
往session/application域对象中共享数据
/** * 测试 向session域中共享数据 * @param session * @return */ @RequestMapping("/test/session") public String testSession(HttpSession session){ session.setAttribute("testSessionScope", "hello,session"); return "success"; } @RequestMapping("/test/application") public String testApplication(HttpSession session){ ServletContext servletContext = session.getServletContext(); servletContext.setAttribute("testApplication","hello,application"); return "success"; }
注意其中的text= ”${session.testSessionScope}”
和text= ”${session.testApplication}”
Spring MVC中的视图
SpringMVC中的视图是View接口,视图的作用渲染数据,将模型Model中的数据展示给用户
SpringMVC视图的种类很多,默认有转发视图和重定向视图 当工程引入jstl的依赖,转发视图会自动转换为JstlView 若使用的视图技术为Thymeleaf,在SpringMVC的配置文件中配置了Thymeleaf的视图解析器,由此视 图解析器解析之后所得到的是ThymeleafView
从图中我们可以看出,所有的我们需要的控制器中的方法都是通过前端显示器DispacherServlet中的反射调用,然后被封装到madelandView对象中
当我们的视图名称中没有任何前缀的话就默认创建的为 themleafView , 如果以forword作为视图前缀的默认就是InternalResourceView (转发视图),如果是以redirect作为视图前缀就默认创建的重定向视图
三种视图:
ThymeleafView :
当控制器方法中所设置的视图名称没有任何前缀时,此时的视图名称会被SpringMVC配置文件中所配置 的视图解析器解析,视图名称拼接视图前缀和视图
后缀所得到的最终路径,会通过转发的方式实现跳转
InternalResourceView :转发视图
SpringMVC中默认的转发视图是InternalResourceView
SpringMVC中创建转发视图的情况:
当控制器方法中所设置的视图名称以”forward:”为前缀时,创建InternalResourceView视图,此时的视 图名称不会被SpringMVC配置文件中所配置的视图解析器解析,而是会将前缀”forward:”去掉,剩余部 分作为最终路径通过转发的方式实现跳转
RedirectView : 默认的重定向视图
当控制器方法中所设置的视图名称以”redirect:”为前缀时,创建RedirectView视图,此时的视图名称不 会被SpringMVC配置文件中所配置的视图解析器解析,而是会将前缀”redirect:”去掉,剩余部分作为最 终路径通过重定向的方式实现跳转
视图控制器 view-controller
作用是:为当前的请求直接设置视图名称,实现页面跳转
RESTful风格
REST:Representational State Transfer,表现层资源状态转移
资源
资源是一种看待服务器的方式,即,将服务器看作是由很多离散的资源组成。
每个资源是服务器上一个 可命名的抽象概念。因为资源是一个抽象的概念,所以它不仅仅能代表服务器文件系统中的一个文件、 数据库中的一张表等等具体的东西,可以将资源设计的要多抽象有多抽象,只要想象力允许而且客户端 应用开发者能够理解。与面向对象设计类似,资源是以名词为核心来组织的,首先关注的是名词。一个 资源可以由一个或多个URI来标识。
URI既是资源的名称,也是资源在Web上的地址。对某个资源感兴 趣的客户端应用,可以通过资源的URI与其进行交互
资源的表述
是一段对于资源在某个特定时刻的状态的描述。可以在客户端-服务器端之间转移(交 换)。资源的表述可以有多种格式,例如HTML/XML/JSON/纯文本/图片/视频/音频等等。资源的表述格 式可以通过协商机制来确定。请求-响应方向的表述通常使用不同的格式。
状态转移
状态转移说的是:在客户端和服务器端之间转移(transfer)代表资源状态的表述。通过转移和操作资 源的表述,来间接实现操作资源的目的。
具体说,就是 HTTP 协议里面,四个表示操作方式的动词:GET、POST、PUT、DELETE。
它们分别对应四种基本操作:GET 用来获取资源,POST 用来新建资源,PUT 用来更新资源,DELETE 用来删除资源。
REST 风格提倡 URL 地址使用统一的风格设计,从前到后各个单词使用斜杠分开,不使用问号键值对方式携带请求参数,而是将要发送给服务器的数据作为 URL 地址的一部分,以保证整体风格的一致性。
由于目前的浏览器只支持get 和post方式的请求,如何发送put和delete请求呢 ?
HiddenHttpMethodFilter (处理delete和put请求)
HiddenHttpMethodFilter 处理put和delete请求的条件:
a> 当前请求的请求方式必须为post
b> 当前请求必须传输请求参数_method,并且value值必须为要处理的put / delete
满足以上条件,HiddenHttpMethodFilter 过滤器就会将当前请求的请求方式转换为请求参数 _method的值,因此请求参数_method的值才是最终的请求方式
在web.xml中设置HiddenHttpMethodFilter
<!-- 设置请求处理的过滤器--> <filter> <filter-name>HiddenHttpMethodFilter</filter-name> <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class> </filter> <filter-mapping> <filter-name>HiddenHttpMethodFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
注意事项
1.在themleaf中,删除操作不能直接将请求参数传进去,必须将请求参数和路径地址进行一下的修改才行
1.进行修改操作时的数据回显,首先要设置input隐藏域,将真正的请求方式写入 and 将id写入,然后对于其他的个个属性进行数据回显
对于单选框(性别等)的数据回显用
在controller层的操作, 先查寻,再进行修改
1.进行删除操作时
要进行确认是否修改
Spring中实现文件上传和下载
文件下载:
ReponseEntity用于控制器方法的返回值
import org.springframework.http.HttpHeaders; import org.springframework.http.HttpStatus; import org.springframework.http.ResponseEntity; import org.springframework.stereotype.Controller; import org.springframework.util.MultiValueMap; import org.springframework.web.bind.annotation.RequestMapping; import javax.servlet.ServletContext; import javax.servlet.http.HttpSession; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; /** * 可以作为一个模板,以后只需要改一些内容即可 */ @Controller public class FileController { @RequestMapping("/test/down") public ResponseEntity<byte[]> testResponseEntity(HttpSession session) throws IOException { //获取ServletContext对象 ServletContext servletContext = session.getServletContext(); //获取服务器中文件的真实路径 String realPath = servletContext.getRealPath("WEB-INF/img"); realPath = realPath + File.separator +"5.png"; System.out.println(realPath); //创建输入流 InputStream is = new FileInputStream(realPath); //创建字节数组 byte[] bytes = new byte[is.available()]; //将流读到字节数组中 is.read(bytes); //创建HttpHeaders对象设置响应头信息 MultiValueMap<String, String> headers = new HttpHeaders(); //设置要下载方式以及下载文件的名字 key为固定的 headers.add("Content-Disposition", "attachment;filename=5.png"); //设置响应状态码 HttpStatus statusCode = HttpStatus.OK; //创建ResponseEntity对象 ResponseEntity<byte[]> responseEntity = new ResponseEntity<>(bytes, headers, statusCode); //关闭输入流 is.close(); return responseEntity; } }
文件上传:
要求:
操作:
创建表单
配置需要的xml文件
<!-- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload --> <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.1</version> </dependency>
开启图片解析器
<!-- 图片解析器--> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"></bean>
上传文件
@RequestMapping("test/up") public String upPhotos(MultipartFile photo,HttpSession session) throws IOException { String filename = photo.getOriginalFilename(); System.out.println(filename); //如果出现重名的情况就做如下操作 String substring = filename.substring(filename.lastIndexOf(".")); String uuid = UUID.randomUUID().toString(); filename = uuid + substring; //这样设置的名字就与其有所差别,不会让文件内容被覆盖 //获取servletContext对象 ServletContext servletContext = session.getServletContext(); //获取当前工程的真实路径 String photoPath = servletContext.getRealPath("photo"); //创建photoPath对应的file对象 File file = new File(photoPath); if (!file.exists()){ //如果没有直接创建 file.mkdir(); } String finalPath = photoPath + File.separator + filename; //上传文件(转移文件到指定的位置) photo.transferTo(new File(finalPath)); return "success"; }
结果(生成图片)