1. 获取 URL 中的参数
获取 URL 中的参数是通过 @PathVariable
注解实现的
"/article/{articleId}") (public String method12( ("articleId") String articleId){ return "接收到参数articleId: " + articleId; }
@RequestMapping("/article/{articleId}")
大括号中的内容就相当于占位符,可以传入任何参数,但是不能不传
多个参数也是可以接受的:
"/article/{articleId}/{name}") (public String method12( String articleId, String name){ return "接收到参数articleId: " + articleId + ", name :" + name; }
前面也提到过,路径中的参数相当于占位符,不能少传,顺序也必须一致
把 required 改为 false 也不行
2. 上传文件
通过 MultipartFile
可以获取文件的对象
"/m12") (public String method13(MultipartFile file){ System.out.println(file.getOriginalFilename()); return "接收到参数file: " + file.getOriginalFilename(); }
通过 Fiddler 抓包可以看到上传的二进制文件
这里的重命名和上面的是不一样的,这里使用的是 @RequestPart
注解
3. 获取 cookie 和 session
3.1. cookie 和 session 的介绍
cookie 就相当于这个令牌,当用户首次访问一个网站时,服务器可以在响应中设置 Cookie,并将其发送给客户端浏览器。浏览器会将 Cookie 存储起来。在后续的请求中,浏览器会自动在请求头中携带该网站的 Cookie 信息发送给服务器。服务器通过读取 Cookie 中的数据来识别用户和获取相关状态信息。
Session 是在服务器端用于跟踪用户会话状态的一种机制,当用户首次与服务器建立连接时,服务器会创建一个唯一的 Session ID,并将这个 ID 返回给客户端。客户端通常会将 Session ID 存储在 Cookie 中(也可以通过 URL 重写等方式传递)。在后续的请求中,客户端会携带 Session ID,服务器通过这个 ID 来识别特定的用户会话,并从服务器端的存储中获取相应的 Session 数据。
上面的学生信息中,每一个 session ID 都对应一个 session 对象
例如,用户在进行网上购物时,将商品添加到购物车。服务器会在 Session 中记录购物车的内容。当用户在不同页面浏览或进行结算时,服务器通过 Session ID 找到对应的购物车数据,确保用户的购物操作在整个会话过程中保持一致。
也就是说,第一次没有 session 时会先创建,然后把 session ID 通过 Set-cookie 来给客户端
session 是保存在内存中的,如果重启服务器 session 数据就会消失丢失
cookie 和 session 的区别:
- cookie 是客户端保存用户信息的一种机制,session 是服务器端保存用户信息的一种机制
- cookie 和 session 之间主要是通过 session ID 关联起来的,session ID 是 cookie 和 session 之间的桥梁
- cookie 和session 经常会配合使用,但是不是必须配合,可以用 cookie 来保存一些数据在客户端,不一定是用户身份信息或 session ID,session 中的 session ID 也不一定必须用 cookie 来保存,还可以通过其他来传递,例如通过 URL 来传递
3.2. 获取 cookie
先来使用创建 HttpServletRequest 对象来调用 getCookies() 的方法来获取 cookie,因为 cookie 还可能为 null,所以也需要进行判断
//获取cookie "getCookie") (public String getCookie(HttpServletRequest request){ Cookie[] cookies = request.getCookies(); if(cookies != null){ for (Cookie cookie : cookies) { System.out.println(cookie.getName() + ":" + cookie.getValue()); } }else{ System.out.println("cookie为空"); } return "获取cookie成功"; }
在 postman 中添加两个 cookie 值,然后获取 cookie
调用 getCookies() 方法来获取 cookie 的方式是获取所有的 cookie,如果需要获取指定的 cookie 的话还需要遍历,加入判断
通过 @CookieValue
注解使用可以直接获取指定名称的 cookie
"getCookie2") (public String getCookie2( ("name") String name){ return "从cookie中获取信息,name:" + name; }
参数名称和 cookie 名是对应的
3.3. 获取 session
由于 getSession 方法是先从 cookie 中找到 session ID,然后根据 session ID 获取 session,所以可能出现 session 为 null 的情况,设置为 true 就会创建一个空的 session,设置为 false 就不会创建,默认是 true
在获取 session 对象之前先设置好 session
"/setSession") (public String setSession(HttpServletRequest request){ //获取session对象 HttpSession session = request.getSession(); session.setAttribute("userName","zhangsan"); session.setAttribute("age",18); return "设置session成功"; } "/getSession") (public String getSession(HttpServletRequest request){ //先从cookie中找到session ID,然后根据session ID获取session HttpSession session = request.getSession(false); if(session == null){ return "获取session为null"; } String userName = (String) session.getAttribute("userName"); return "从session中获取userName: " + userName; }
浏览器中先访问 getSession 时,由于还没有设置 session,所以此时还获取不到 session 信息,不过由于用的是 getSession() 方法,如果 session 为空时会创建一个 session,
设置之后就可以正常获取了
如果换个浏览器重新获取就又获取不到了
这就是因为处于不同的会话中,会话的 id 是不同的,这也就是相当于不同用户访问时处于不同的 session 中会获取不同的 session ID
也可以直接传入 HttpSession 对象来获取:
"/getSession2") (public String getSession2(HttpSession session){ String userName = (String) session.getAttribute("userName"); if(userName !=null){ return "从session中获取userName: " + userName; }else { return "获取session为null"; } }
还有一种方式获取就是通过 @SessionAttribute
注解来获取
"/getSession3") (public String getSession3( String userName){ return "从session中获取userName: " + userName; }
通过这种方式来获取指定 session ,这个 session 是必须要设置好的,如果不存在的话就会直接报错,要求强制绑定,例如上面的 userName 如果找不到就会报错
4. 获取 header
http 的请求头是有很多个键值对的,可以通过创建 HttpServletRequest 对象来调用 getHeader 方法来获取指定的请求头
"/getHeader") (public String getHeader(HttpServletRequest request){ String User_Agent = request.getHeader("User-Agent"); return "User_Agent: " + User_Agent; }
还可以通过 @RequestHeader
注解来获取
"/getHeader2") (public String getHeader2( ("User-Agent") String User_Agent){ return "User_Agent: " + User_Agent; }
5. 响应
5.1. 返回静态页面
先创建一个 html 页面
如果还按照之前的方式进行返回的话,返回的并不是一个 html 页面
"/response") ( public class ResponseController { "/returnHtmlPage") ( public String returnHtmlPage(){ return "/hello.html"; } }
如果把 @RestController
改成 @Controller
就可以返回 html 页面
"/response") ( public class ResponseController { "/returnHtmlPage") ( public String returnHtmlPage(){ return "/hello.html"; } }
对比这两个注解的实现可以发现,可以理解为@RestController
是在 @Controller
的基础上实现的,相差了一个 @ResponseBody
@ResponseBody
表示返回数据,如果使用 @Controller
再加上@ResponseBody
就又显示的是数据了
@ResponseBody
既是类注解又是方法注解,如果作用在类上,表示该类的所有方法都返回的是数据,如果作用在方法上,表示该方法返回的是数据,其它的不受影响
此外,由于 @Controller
返回的是页面,所以用到了像之前使用的 @RestController
返回数据的代码中,就可能报错
如果把原来的返回值当成页面路径返回后,就找不到这个页面,就会给出 404 的状态码
5.2. 返回 html 代码片段
如果还是用 @Controller
的话,由于返回的是页面,就找不到对应的路径,所以需要加上 @ResponseBody
,直接写上去的 html 代码片段会被自动识别:
"/returnHtml") (public String returnHtml(){ return "<h1>你好</h1>"; }
5.3. 返回 JSON
JSON 是可以表示对象的,返回对象时就会通过 JSON 方式返回
"/returnJson") (public User returnJson(){ User user = new User(); user.setName("zhangsan"); user.setAge(18); return user; }
这里的返回类型也变为了 JSON
5.4. 设置状态码
状态码可以通过创建 HttpServletResponse 对象调用 setStatus 方法进行设置,例如设置一个 404
"/setStatus") (public User setStatus(HttpServletResponse response){ User user = new User(); user.setName("zhangsan"); user.setAge(18); response.setStatus(404); return user; }
虽然说显示的是 404 状态码,但是还是显示了页面
5.5. 设置 header
在 @RequestMapping
中设置一些参数可以修改指定 header 的一些类型
来修改一下请求类型:
value = "/setHeader",produces = "application/json") (public String setHeader(){ return "<h1>你好</h1>"; }
还可以添加一个 Header 信息
value = "/setHeader2") (public String setHeader2(HttpServletResponse response){ response.setHeader("name","zhangsan"); return "success"; }