题目
页面StudentLogin.html中有一HTML的表单代码如下:
<form action="studentLogin" method="post">学生姓名:<input type="text" name="stuName" value=""><br>登录密码:<input type="password" name="stuPwd" value=""><br><input type="submit" name= "" value="登录">
</form>
现按下面的要求定义一个Servlet:
1)定义一个Servlet,将StudentLogin.html的表单数据提交到这个Servlet,根据StudentLogin页面表单中的设置来配置本Servlet的访问路径,要求采用注解方式配置访问路径;
2)本Servlet中获取StudentLogin.html表单提交的学生姓名、登录密码,若姓名为“赵钱孙”,且密码为“123456”,则将该学生姓名添加为request对象以实现数据流转,再将请求内部派发home.jsp;反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;
3)设置输出Servlet的响应数据类型为html。
注释:
1)设置request作用域:
request.setAttribute(String,Object);
2)页面请求内部派发:
request.getRequestDispatcher(url).forward(request,response);
3)3秒后页面自动跳转
response.setHeader("Refresh","3;URL=路径");
注意:只用编写Servlet文件,不用编写jsp和html文件。
本章知识重点:
目录
HTML里的表单元素
解读表单代码:###
1.定义Servlet:
1.2 方便验证——创建一个html文件 ##
1.3 编写Servlet_注解方式配置Servlet路径
2. doPost方法重写(重载)#####
2.1 分析题意: #####
2.2 设置request作用域:###
请求作用域是何物
2.3 将请求内部派发:
2.4 3秒后页面自动跳转
完整代码参考 :
前文链接:JAVAWeb_Servlet初识与代码入门-CSDN博客
JAVAWeb_Servlet初识与代码入门-CSDN博客
我们本节重中之重,便是学会处理POST请求。但是先来看“题目”一上来,给了我们一段代码——“StudentLogin.html中有一HTML的表单代码”。这段代码是什么,对于我们处理的POST请求有什么关系?
HTML里的表单元素
<form ......>......
</form>
<form>一出现,而</form>一结尾(体现标签成对出现的特点),就表示两个标签中间的代码是表单代码。
表单代码一般放在HTML文件里。因为:HTML是一种用于构建网页的标记语言,它包含多种元素,用于定义网页的结构、内容和布局。表单只是HTML中众多元素之一,用于收集用户输入的数据并提交到服务器。
表单代码收集用户数据?这就很妙了,请求正是要收集(服务器所需的)用户数据:GET请求里我们通过URL的查询字符串的方法收集用户数据;POST请求里我们就用表单收集用户数据。
解读表单代码:
接下来,我们通过解读这段表单代码来了解表单代码怎么收集用户信息。
<form action="studentLogin" method="post">...
</form>
<form>
标签成对出现。所以一头为<form>,末尾</form>。
我们再着重看第一行:
<form action="studentLogin" method="post">
// 在写form代码时,就要规定action和method,这是<form> 标签的属性,它们定义了表单提交时的行为。
// action="studentLogin" ,指定表单数据提交的目标URL ——因为这里的表单数据不是为了收集而收集,而是为了交给服务器处理。服务器怎么处理,自然是交给能处理的POST请求的Servlet。所以这个URL,正是Servlet的URL映射路径,表单收集到的数据都会通过URL,找到并交给能处理这段数据的Servlet。//指定表单数据提交的方式为HTTP POST方法。POST方法通常用于提交敏感数据(如登录信息),因为它不会将数据暴露在URL中。
若读者通过了GET请求那一题,便知道GET请求传数据是直接在URL里传参数的键值对(可谓是光明正大,密码都是通过明文并且不加修饰直接交给Servlet。客户敢用,我们都不敢这么写)。
现在来看:<form>标签里的代码,为了更好理解,我们结合运行.html文件出的表单。
学生姓名:<input type="text" name="stuName" value=""><br>
登录密码:<input type="password" name="stuPwd" value=""><br><input type="submit" name= "" value="登录">
先来直观理解<form>生成的表单内容:
可以看到:
①:“学生姓名:”就直接显示,后面"<input ......>"对应了网页里生成的输入框。
②:“登录密码:”也直接显示,后面"<input ......>"对应了网页里生成的输入框。
③:因为只有<input......>它其实也是输入框,不过它已经被赋值了“登录”。所以这个框直接显示“登录”。
这里大家会注意新的标签<input>:介绍如下
<input type="text" name="username" value="default value"><br>
//<input>也有三个关键属性:在定义时会让我们初始化
// 1. type 属性定义了输入字段的类型。text:普通文本输入框,用户可以输入任意文本。
// 2. name 属性为输入字段指定一个名称,这个名称在表单提交时用于标识该输入字段(说人话,就是该输入框的名称&&参数的键值对中的“键”)
// 3. value 属性定义了输入字段的默认值或当前值。(说人话,就是输入框里的数据 && 参数键值对中的“值”)// <br>是换行
那么此处这三句代码就好理解了:
学生姓名:<input type="text" name="stuName" value=""><br>
// 学生姓名:直接打印输出在页面
// <input> 为HTML输入元素,用于创建一个输入框
// type="text" 表示为一个文本输入框,用户可以在其中输入任意文本。
// name="stuName" 为输入框指定一个名称,服务器端可以通过这个名称来获取对应的值
// value="" 指定输入框的默认值。这里默认值为空,表示用户需要手动输入学生姓名。
登录密码:<input type="password" name="stuPwd" value=""><br>
// 这是一个密码输入框,类型(type)为"password"。其特点:输入时会以隐藏形式显示(通常显示为圆点或星号),以保护用户输入的敏感信息。
// 输入框名称为stuPwd (键值对中的键)
// 默认输入值为空,等待学生输入密码 (键值对中的值)
<input type="submit" name= "" value="登录">
//这个输入框,前面就不显示什么文本信息了。
//类型为"submit",name=""为按钮指定一个名称。这里名称为空,表示该按钮没有特定的名称。
// 但这个按钮(输入框)默认值为“登录”,于是在网页上就会在这个按钮上浮现这个值
// submit类型的输入框:用户点击后触发表单提交操作。
也就是说,提交的表单包含了两个参数对(学生姓名 和登录密码 与其对应的值们) 当点击submit类型的按钮,就会提交表单。表单会提交到我们编写的servlet里(因为收集用户端数据,就是为了交给服务器端处理)。为了处理这种以表单形式收集数据的请求,我们将这种请求称为POST请求,于是Servlet编写的处理请求的方法—— 正是doPost(一种专门处理POST请求)方法。
1.定义Servlet:
接下来,我们开始写代码。
首先得创建JAVA Web项目,作者这里就创建好了:
接着需要新建一个Servlet文件:
Servlet名字也没规定,作者自定义了。
Eclipse生成的代码,我删了些(锻炼代码能力嘛)
1.2 方便验证——创建一个html文件
如标题所示,就是为了方便验证。接下来展示怎么创建一个html文件。
自动生成的HTML File具有以下代码:
这里我们需要编写的HTML代码是一段表单,因为题上已经给出,我们直接copy上去(呵呵)
(忽略.html的名字,就按照题意取名,作者是后面发现不对劲的 T_T)
1.3 编写Servlet_注解方式配置Servlet路径
使用注解方式配置Servlet,我们都会采用@WebServlet(),这是前面提过的以及Servlet的代码开头、整体框架,我就直接开头喽~
@WebServlet("/")
public class stuLoginservlet extends HttpServlet
{// 继承HttpServlet,是为了什么?为了保证自己写的Servlet在语法上实现Servlet所有方法;语法上实现,意味着我们开发者不需要写那些代码// 而又可以继承HttpServlet提供的doGet和doPost方法,根据我们的需要进行改写
}
类名须于.java文件名一致。
我们现在来思考:用户可以通过哪个URL访问到我们的Servlet,这也是@WebServlet注释的作用。
再看一眼题意:
根据我们前面“表单代码” 以及这里不断明示,为Servlet配置的路径,正是表单要提交的URL("studentLogin"),而表单提交的方式"method = "post"",也意味着这份表单要交给能处理post请求的servlet。
@WebServlet("/studentLogin")//表单根据action的值找到g该值映射的这个servlet
public class stuLoginservlet extends HttpServlet
{//为了处理post请求,我们得改写doPost方法protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{}
}
2. doPost方法重写(重载)
2.1 分析题意:
在分析题意中,我们会把整体代码逻辑和能写的代码完成~
本Servlet获取StudentLogin.html表单提交的学生姓名、登录密码
很明显,我们需要先获取参数,没问题—— request.getParameter();
String name = request.getParameter("stuName");//传进去的参数,一定是参数对的键。因为要获取用户输入的值:这里获取的是stuName的值
String pwd = request.getParameter("stuPwd");//这里pwd获取的是:用户对应stuPwd这个输入框的值
“若则”关系,那么需要使用if-else :
//获取参数的值
String name = request.getParameter("stuName");
String pwd = request.getParameter("stuPwd");
//若则关系
if()
{}
else
{}
//若姓名为"赵钱孙",密码为"123456"
if("赵钱孙".equals(name) && "123456".equals(pwd))
//String.equals(String) 判断两个字符串是否相等,第一个String不能为null否则会抛出异常
接下来写伪代码:
//获取参数的值
String name = request.getParameter("stuName");
String pwd = request.getParameter("stuPwd");
//若则关系
if("赵钱孙".equals(name) && "123456".equals(pwd))
{//将该学生姓名添加为request对象以实现数据流转,再将请求内部派发home.jsp
}
else
{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;
}
在浏览器上显示页面信息?这我们学过啊,直接写:
//在浏览器上显示信息//1.获取响应输出流
PrintWriter out = response.getWriter();
//2.输出信息
out.println("登录失败,3秒钟后请重新登录");
//获取参数的值
String name = request.getParameter("stuName");
String pwd = request.getParameter("stuPwd");
//若则关系
if("赵钱孙".equal(name) && "123456".equal(pwd))
{//将该学生姓名添加为request对象以实现数据流转,再将请求内部派发home.jsp
}
else
{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;PrintWriter out = response.getWriter();out.println("登录失败,3秒钟后请重新登录");
}
其他的“内部派发”,"自动跳转",我们还没学,但是代码逻辑基本完整,后续补充一下就能上手。还有些细节需要处理:
3)设置输出Servlet的响应数据类型为html。
//这也是我们学习过的—— response.setContentType();
response.setContentType("text/html;charset=UTF-8");
现在来综合我们写的代码,来发现还有哪些还没解决:
@WebServlet("/studentLogin")
public class stuLoginservlet extends HttpServlet
{protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{//以后都要习惯,只要涉及到中文;不管题上是否让响应输出html格式,都得第一步设置响应的编码格式// 1.设置编码response.setContentType("text/html;charset=UTF-8");// 2. 获取请求参数的值String name = request.getParameter("stuName");String pwd = request.getParameter("stuPwd");// 3. 判断请求参数值if("赵钱孙".equals(name) && "123456".equals(pwd)){//将该学生姓名添加为request对象以实现数据流转,再将请求内部派发home.jsp}else{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;PrintWriter out = response.getWriter();out.println("登录失败,3秒钟后请重新登录");}}
}
还没解决的问题:
1. 对于请求体的编码,服务器并没设置编码—— 如果请求体参数的值为中文,服务器默认的编码是不支持识别中文。
2. 将该学生姓名添加为request对象以实现数据流转,再将请求内部派发home.jsp
3. 在3秒后自动跳转StudentLogin.html页面;
对于第1点,我们只需熟悉(掌握)这句话:
request.setCharacterEncoding("UTF-8");
//处理表单数据,一定要设置请求体的编码,确保服务器能正确解析传来的字符(中文)
所以处理Post请求,一定二话不说,方法体的第一步是两个编码:
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
还剩最后两点:
出题人也是很善良体贴了,都贴上了相应的方法,所以这篇作者会重点讨论第一个方法,后面两个方法直接考虑怎么用就完了(锻炼我们现学现卖的技能):
2.2 设置request作用域:
方法介绍(参考官方文档格式)
public void request.setAttribute(String name,Object obj);
即 request.setAttribute(String name,Object obj);
方法描述
将一个对象obj与指定的名称name绑定到当前请求的作用域中。此方法允许将数据存储
HttpServletRequest
对象(此处为doPost里的request)中,以便在同一个请求的后续处理过程中访问这些数据。例如,可以将数据传递给后续的Servlet或JSP页面。(此处题意为传给home.jsp)
参数
name
:要绑定到请求作用域的属性名称。该名称必须是String
类型,且不能为null。
o
:要绑定到请求作用域的属性值。该值可以是任何Object
类型,包括null
异常
IllegalArgumentException
:如果name
为null
。
注意:
· 作用域限制:
使用
setAttribute
方法设置的属性仅在当前请求的作用域中有效。一旦请求结束,这些属性将不再可用。· 覆盖属性:
如果已经存在一个与指定名称相同的属性,调用
setAttribute
方法将覆盖原来的属性值。
请求作用域是何物
请求作用域 是Web应用程序中的一种数据存储范围,它与单个HTTP请求相关联。好理解,这个作用域就是存数据的。
为什么需要请求作用域。(即它的作用):刚刚介绍的设置request的作用域方法,结合此处所说的请求作用域。就知道设置request的作用域实际上就是把一对参数(它的名称和值)存进请求作用域。从题意上,我们其实能窥见请求作用域的作用——将数据传递给后续的Servlet或JSP页面
(作者觉得这出题人这里描述得有问题,更恰当地——“将学生姓名添加到request对象”)
更加完整的作用:
数据共享——在处理一个HTTP请求时,可能需要在多个组件(如Servlet、JSP页面)之间共享数据。请求作用域提供了一个方便的地方来存储这些数据,使得它们可以在请求的生命周期内被访问。(重点)
简化数据传递——通过将数据存储在请求作用域中,可以避免在组件之间手动传递数据。例如,从Servlet到JSP页面,或者从一个Servlet到另一个Servlet。
于是,结合分析,代码如下:
request.setAttribute("StuName",name);//设为request作用域的对象之一,又是一个键值对啊盆油:键作者自己取了,值为我们获取到的参数值name
@WebServlet("/studentLogin")
public class stuLoginservlet extends HttpServlet
{protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{// 1.设置编码request.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");// 2. 获取请求参数的值String name = request.getParameter("stuName");String pwd = request.getParameter("stuPwd");// 3. 判断请求参数值if("赵钱孙".equal(name) && "123456".equal(pwd)){//将该学生姓名添加为request对象以实现数据流转request.setAttribute("StuName",name);//再将请求内部派发home.jsp}else{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;PrintWriter out = response.getWriter();out.println("登录失败,3秒钟后请重新登录");}}
}
2.3 将请求内部派发:
2)页面请求内部派发:
request.getRequestDispatcher(url).forward(request,response);
很明显,若想实现“再将请求内部派发home.jsp”,我们只要把这句代码copy上去,把参数正确填写就能实现:
request.getRequestDispatcher("/home.jsp").forward(request,response);
// url 是一个字符串,表示目标资源的路径。推荐加“/”,绝对路径写法
//后面的request,response不用改变,我们当前servlet的request,response就是这个名字
2.4 3秒后页面自动跳转
3)3秒后页面自动跳转
response.setHeader("Refresh","3;URL=路径");
题目要求:3s后页面自动跳转到 StudentLogin.html 页面。只需要替换url的值就行(👌)
response.setHeader("Refresh","3;URL=StudentLogin.html");
我们所有代码都已经完成:
@WebServlet("/studentLogin")
public class stuLoginservlet extends HttpServlet
{protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{// 1.设置编码request.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");// 2. 获取请求参数的值String name = request.getParameter("stuName");String pwd = request.getParameter("stuPwd");// 3. 判断请求参数值if("赵钱孙".equals(name) && "123456".equals(pwd)){//将该学生姓名添加为request对象以实现数据流转request.setAttribute("StuName",name);//再将请求内部派发home.jsprequest.getRequestDispatcher("/home.jsp").forward(request,response);}else{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;PrintWriter out = response.getWriter();out.println("登录失败,3秒钟后请重新登录");response.setHeader("Refresh","3;URL=StudentLogin.html");}}
}
验证:
显示:
重新跳转:
另一种情况:
没事儿,现在保证自己写的代码逻辑写对就行。我们本来也没写home.jsp ,后面我们揭晓jsp。
完整代码参考 :
@WebServlet("/studentLogin")
public class stuLoginservlet extends HttpServlet
{protected void doPost(HttpServletRequest request,HttpServletResponse response) throws ServletException,IOException{// 1.设置编码request.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");// 2. 获取请求参数的值String name = request.getParameter("stuName");String pwd = request.getParameter("stuPwd");// 3. 判断请求参数值if("赵钱孙".equals(name) && "123456".equals(pwd)){//将该学生姓名添加为request对象以实现数据流转request.setAttribute("StuName",name);//再将请求内部派发home.jsprequest.getRequestDispatcher("/home.jsp").forward(request,response);}else{//反之则显示“登录失败,3秒钟后请重新登录”,并在3秒后自动跳转StudentLogin.html页面;PrintWriter out = response.getWriter();out.println("登录失败,3秒钟后请重新登录");response.setHeader("Refresh","3;URL=StudentLogin.html");}}
}
经过这两篇文章的铺垫,对于定义基本Servlet我们至少不会再手足无措——多敲多敲,希望读者(包括作者)能在定义Servlet时得心应手(放心,后面还有练习,作者补充知识的篇幅就不这么大了)。
前文链接:JAVAWeb_Servlet初识与代码入门-CSDN博客
JAVAWeb_Servlet初识与代码入门-CSDN博客
作者有话说:(我真没想到能破万(扶额邪魅一笑),嘶~为了咱的读者后期好查阅,我会将一些篇幅大的重要部分单拎出来(点头))
不过能看到这里的读者,作者也要为你点赞👍