【Spring 】Spring MVC 入门Ⅱ

Spring MVC 入门Ⅱ

一、接收Cookie / Session

  1. 这两者都是用来保存用户信息的,但不同的是:

    1. Cookie存在客户端在这里插入图片描述
    1. Session存在服务器
  2. Session产生时会生成一个唯一性的SessionID,这个SessionID可以用于匹配Session和Cookie

  3. SessionID可以在Cookie/Set-Cookie传递,也可以在URL中传递


1. 获取Cookie(获取所有的):

代码:

    @RequestMapping("/r12")public String r12(HttpServletRequest request, HttpServletResponse response) {// 获取CookieCookie[] cookies = request.getCookies();// 打印StringBuilder builder = new StringBuilder();if (cookies != null) {for (Cookie cookie : cookies) {builder.append("key: "+cookie.getName() + "  value:" + cookie.getValue()+"\n");}}return "Cookie信息:" + builder;}

结果:

在这里插入图片描述

从这个示例中可以看出,Cookie是可以伪造的,所以后端在接收Cookie信息时需要进行Cookie校验

2. 简洁获取Cookie

可以通过@CookieValue注解进行指定cookie的key值进行获取value。

代码:

    @RequestMapping("/r13")public String r13(@CookieValue("name") String name) {return "r13获取到cookie:" + name;}

结果:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

二、获取Session

1. 理解Session

Session是用于存储用户信息并且保存在服务器端的数据。这个过程是无状态的。

什么是会话?

会话就是客户端与服务器进行的一次建立连接发送数据的过程。

就像在店铺咨询客服一样,每次发起一次咨询的过程就是一次会话。

什么是无状态的

每次发起新的咨询,即使是同一个客服对你进行服务,这个客服仍然不知道你是谁。这个过程是无记忆的,断开连接就“恢复出厂设置了”


2. 获取Session

2.1HttpServletRequest获取
package com.example.springmvcdemo;import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import jakarta.servlet.http.HttpSession;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/session")
public class SessionController {@RequestMapping("/setSession")public String setSession(HttpServletRequest request, HttpServletResponse response) {HttpSession session = request.getSession();session.setAttribute("name", "zhangsan");return "session设置成功";}@RequestMapping("/getSession")public String getSession(HttpServletRequest request) {HttpSession session = request.getSession();String name = (String) session.getAttribute("name");return "name: "+name;}
}

一定要先进行设置Session,再获取Session。

在这里插入图片描述

2.2 HttpSession获取
@RequestMapping("/getSession2")public String getSession2(HttpSession session) {return "HttpSession获取Session成功:"+session.getAttribute("name");}

在这里插入图片描述在这里插入图片描述

2.3 @SessionAttribute获取
 @RequestMapping("/getSession3")public String getSession2(@SessionAttribute("name") String name) {return "@SessionAttribute获取Session成功:"+name;}

在这里插入图片描述
在这里插入图片描述

三、获取Header

3.1 HttpServletRequest获取

代码:

@RestController
@RequestMapping("/header")
public class HeaderController {@RequestMapping("getHeader")public String getHeader(HttpServletRequest request) {String header = request.getHeader("User-Agent");return "HttpServletRequest获取到Header:"+header;}
}

结果:
Fiddler抓包:
在这里插入图片描述

3.2 @RequestHeader获取

代码:

    @RequestMapping("getHeader2")public String getHeader(@RequestHeader("User-Agent") String ua) {return "@RequestHeader获取到Header:"+ua;}

结果:

在这里插入图片描述


3.3 设置Header
3.3.1 设置Content-Type

代码:

@RequestMapping(value = "setHeader", produces = "application/json")public String setHeader(HttpServletRequest request) {return "{key:value}";}

在这里插入图片描述

3.3.2 设置状态码

代码:

    @RequestMapping(value = "setStatus")public String setStatus(HttpServletResponse response) {int status = 345;response.setStatus(status);return "状态码设置成功: " + status;}

结果:

在这里插入图片描述

3.3.3 设置User-Agent

代码:

    @RequestMapping(value = "setHeader2", produces = "application/json")public String setHeader2(HttpServletResponse response) {response.setHeader("User-Agent", "abc");return "{key:value}";}

结果:

在这里插入图片描述

3.3.4 添加Header信息

代码:

    @RequestMapping(value = "setHeader3", produces = "application/json")public String setHeader3(HttpServletResponse response) {response.setHeader("MyHeader", "2024qyy");return "{key:value}";}

结果:

在这里插入图片描述

四、两个Demo

4.1 计算器

4.1.1 后端接口

响应数据:两数之和

数据类型:text

4.1.2 前端接口

请求路径:user/login

请求方式:get/post

接口描述:计算两数之和

请求参数:

参数描述类型是否必传
num1加数1Integer
num2加数2Integer

响应数据:两数之和

4.1.3 后端代码
package com.example.springmvcdemo.controller;import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RequestMapping("/calc")
@RestController
public class CalcController {@RequestMapping("sum")public String sum(Integer num1, Integer num2) {Integer sum = num1+num2;return "两数之和:"+ sum;}
}

4.2 用户登录

4.2.1 后端接口

登录接口:

响应数据:登录是否成功

数据类型:Boolean

获取用户名接口:

响应数据:用户名

数据类型:text

4.2.2 前端接口

登录接口:

请求路径:user/login

请求方式:post

接口描述:验证用户登录是否成功

请求参数

参数类型说明是否必传
userNameString用户名
passwordString密码

获取用户名接口:

请求路径:user/index

请求方式:get

接口描述:获取用户名

请求参数:不需要参数

4.2.3 后端代码
package com.example.springmvcdemo.controller;import jakarta.servlet.http.HttpSession;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.bind.annotation.SessionAttribute;@RestController
@RequestMapping("/user")
public class UserController {/*** 用户登录* @param userName* @param password* @param session* @return*/@RequestMapping("/login")public Boolean login(String userName, String password, HttpSession session) {if (!StringUtils.hasLength(userName) || !StringUtils.hasLength(password)) {return false;}System.out.println("接收到用户名:"+userName);System.out.println("接收到密码:"+password);if (!"admin".equals(userName) || !"password".equals(password)) {return false;}// 保存到session中——key一定要带引号session.setAttribute("userName", password);return true;}/*** 获取登录用户名* @param userName 之前保存在Session中的用户名,进行取出* @return*/@RequestMapping("/index")public String getUserName(@SessionAttribute("userName") String userName) {return userName;}
}
4.2.4 前端代码:
<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><title>登录页面</title>
</head><body><h1>用户登录</h1>用户名:<input name="userName" type="text" id="userName"><br>密码:<input name="password" type="password" id="password"><br><input type="button" value="登录" onclick="login()"><script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.4/jquery.min.js"></script><script>function login() {$.ajax({type: "post",url: "/user/login",data:{userName:$("#userName").val(),password:$("#password").val(),},success:function (result) {if (result) {location.href = "/index.html";alert("跳转成功!");}else {alert(JSON.stringify(userName));alert(JSON.stringify(password));alert("用户名或密码错误");}}});}</script>
</body></html>
<!doctype html>
<html lang="en"><head><meta charset="UTF-8"><meta name="viewport"content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"><meta http-equiv="X-UA-Compatible" content="ie=edge"><title>用户登录首页</title>
</head><body>登录人: <span id="loginUser"></span><script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.4/jquery.min.js"></script><script>$.ajax({type:"get",url:"/user/index",success:function (userName) {$("#loginUser").text(userName);}});</script>
</body></html>

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

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

相关文章

模型训练中的过拟合和欠拟合

基本概念 我们知道&#xff0c;所谓的神经网络其实就是一个复杂的非线性函数&#xff0c;网络越深&#xff0c;这个函数就越复杂&#xff0c;相应的表达能力也就越强&#xff0c;神经网络的训练则是一个拟合的过程。   当模型的复杂度小于真实数据的复杂度&#xff0c;模型表…

python中的进程线程和协程

目录 进程&#xff08;Process&#xff09;多进程代码实例 线程&#xff08;Thread&#xff09;多线程存在原因及其缺点多线程代码实例 协程&#xff08;Coroutine&#xff09;协程的优点协程代码实例 进程、线程和协程适合的任务性质和环境多进程更适合的场景多线程更适合的场…

智能优化算法--计算重复运行50次的最大值、最小值、均值、标准差

⚠申明&#xff1a; 未经许可&#xff0c;禁止以任何形式转载&#xff0c;若要引用&#xff0c;请标注链接地址。 全文共计3077字&#xff0c;阅读大概需要3分钟 &#x1f308;更多学习内容&#xff0c; 欢迎&#x1f44f;关注&#x1f440;【文末】我的个人微信公众号&#xf…

【ARM Cache 系列文章 11 -- ARM Cache 直接映射 详细介绍】

请阅读【ARM Cache 系列文章专栏导读】 文章目录 ARM Cache组织形式直接映射(Direct Mapped)直接映射示例直接映射原理Cache颠簸(cache thrashing)原因文章:【ARM Cache 系列文章 11.1 – ARM Cache 全相连 详细介绍】 文章:【ARM Cache 系列文章 11.2 – ARM Cache 组相…

在Android中,如何通过Kotlin协程处理多个API调用

在Android中&#xff0c;如何通过Kotlin协程处理多个API调用 在Android开发中&#xff0c;如何使用Kotlin协程处理多个API调用的示例呢&#xff1f;假设我们已经对Kotlin协程有了一定的了解&#xff0c;包括定义、简单用例和示例等。现在&#xff0c;让我们来看一些真实的Andr…

Tokitsukaze and Average of Substring

原题链接&#xff1a;登录—专业IT笔试面试备考平台_牛客网 目录 1. 题目描述 2. 思路分析 3. 代码实现 1. 题目描述 2. 思路分析 前缀和。 开一个int类型的前缀和数组pre[30][N]&#xff08;pre[i][j]表示某字符转成的数字 i 在一段区间的前缀个数。因为字母表有‘a’~z…

带你学C语言:结构体及其内存

目录 &#x1f37a;0.前言 ✍1.结构体 &#x1f440;1.1为何结构体 &#x1f440;1.2结构体怎么声明 &#x1f440;1.3结构体怎么创建 &#x1f440;1.4结构体初始化与访问 ✋1.5匿名结构体问题 &#x1f646;1.6结构体的自我调用 &#x1f69d; 2.结构体的内存对齐 &a…

【数据结构】时间复杂度和空间复杂度解析

数据结构前言&#xff1a; 1. 什么是数据结构 打个比方来说不同的数据就相当于不同的书籍&#xff0c;我们经常在图书馆可以看到不同类别的书籍会被整理放在书架上方便查看存放&#xff0c;数据结构就是一种计算机存储管理数据的方式。 2. 什么是算法 算法就是一系列的计算…

FlinkSQL 中lateral table

在 Flink SQL 中&#xff0c;LATERAL TABLE 是一种用于处理依赖于外部表达式的表值函数&#xff08;Table-valued Function&#xff0c;简称 TVF&#xff09;的语法。LATERAL TABLE 用于在查询中扩展表&#xff0c;并将表值函数的结果与查询的其余部分进行连接&#xff08;LATE…

UDP和TCP(传输层)

这里写目录标题 UDPUDP的基本特点UDP协议报文格式 TCPTCP协议报文格式TCP特点可靠传输实现机制确认应答超时重传数据丢了应答报文丢了 小结 UDP UDP的基本特点 无连接不可靠传输面向数据报全双工 UDP协议报文格式 2个字节有效范围(无符号): 0 ~ 65535(2^16 - 1). 2个字节有效范…

笨蛋学C++之 C++对数据库实现CRUD

笨蛋学C 之 C对数据库实现CRUD 头文件testcrud.h 源文件testcrud.cppmain.cpp 头文件 testcrud.h #pragma once #include <mysql.h> #include <iostream> #include <vector> #include <cstring> // 包含字符串操作相关的头文件 using namespace std;…

安装 AngularJS

安装 AngularJS 文章目录 安装 AngularJS1. 使用在线 cdn2. 使用依赖管理工具 npm 1. 使用在线 cdn <!-- 1. 引入在线地址 --> <script src"http://code.angularjs.org/1.2.25/angular.min.js"></script><!-- 2. 下载到本地&#xff0c;引入文…

C语言双向链表如何实现向指定索引位置插入元素

核心代码&#xff1a; int insertDoubleLinkedList(DoubleLinkedList *list, int index, int value) {if (list NULL || list->head NULL || list->tail NULL) {return -1;}// 当一个元素都没有的时候&#xff0c;或者indexsize的时候&#xff0c;是支持插入的if (in…

前端html中iframe的基本使用

以下是一个比较复杂的 <iframe> 示例&#xff0c;展示了如何加载外部页面、控制样式和与 <iframe> 中加载的文档进行通信&#xff1a; <!DOCTYPE html> <html> <head><style>.iframe-container {position: relative;width: 100%;height: …

【Python】常用数据结构

1、熟悉字典和列表 2、使用条件判断语句 3、list列表中计算 1、从键盘输人一个正整数列表,以-1结束,分别计算列表中奇数和偶数的和。 &#xff08;1&#xff09;源代码&#xff1a; # 初始化奇数和偶数的和为0 odd_sum 0 even_sum 0 #输入 while True:num int(input(&qu…

Android学习系列目录

Android学习1 -- 从嵌入式Linux到嵌入式Android-CSDN博客 Android学习2 -- SDK 1&#xff08;概览&#xff09;-CSDN博客 Android学习3 -- SDK2 &#xff08;实操三个小目标&#xff09;-CSDN博客 Android学习4 -- ADB的使用-CSDN博客 Android学习5 -- HAL-1 概述-CSDN博客…

ubuntu下安装配置python3.11

方案1 添加仓库&#xff1a; $ sudo add-apt-repository ppa:deadsnakes/ppa $ sudo apt update $ sudo apt install python3.11然后查看有多少个python版本已经安装了&#xff1a; ls -l /usr/bin/python*python2.7,python 3.8 ,python 3.11. 然后&#xff0c;设置系统默认…

智能车入门——‘教程引导’ <新手从零做车>

目录 前言 本系列文章是为了帮助第一次接触智能车或者学校没有传承&#xff0c;不知道如何上手做智能车的同学。 通过阅读完整个系列&#xff0c;你应该能够制作一辆正常参赛的智能车。 我写这一系列博客的初衷主要是为了方便新手快速入门智能车。 如果追求高级算法以及提速&a…

【Jenkins】持续集成与交付 (七):Gitlab添加组、创建用户、创建项目和源码上传到Gitlab仓库

🟣【Jenkins】持续集成与交付 (七):Gitlab添加组、创建用户、创建项目和源码上传到Gitlab仓库 1、创建组2、创建用户3、将用户添加到组中4、在用户组中创建项目5、源码上传到Gitlab仓库5.1 初始化版本控制5.2 将文件添加到暂存区5.3 提交代码到本地仓库5.4 推送代码到 Git…

【如何成功安装 Python 软件包 weditor】

如何成功安装 Python 软件包 weditor 在进行软件开发或者使用 Python 进行编程时&#xff0c;经常会遇到需要安装第三方软件包的情况。然而&#xff0c;有时候安装过程并不顺利&#xff0c;可能会遇到各种问题。在本文中&#xff0c;我将分享我解决安装 Python 软件包 weditor…