目录
- 一、简介
- 二、`Servlet`入门案例
- 2.1、编写`Servlet`
- 2.2、配置`Servlet`
- 2.3、访问`Servlet`
- 2.4、常见错误
- 三、`Servlet`详解
- 3.1、实现`Servlet`的三种方式
- 3.1.1、实现`Servlet`接口
- 3.1.2、继承`GenericServlet`类
- 3.1.3、继承`HttpServlet`类
- 3.2、配置`Servlet`的两种方式
- 3.2.1、`web.xml`方式
- 3.2.2、注解方式
- 3.3、常见问题
- 3.4、关于`Jrebel`
- 3.4.1、安装
- 3.4.2、激活
- 3.4.3、使用之前的配置
- 3.4.4、启动测试
- 四、`request`和`response`
- 4.1、请求响应流程
- 4.2、`response`对象
- 4.2.1、设置响应正文
- 4.2.2、响应乱码问题处理
- 4.2.3、重定向
- 4.3、`request`对象
- 4.3.1、获取请求头(了解)
- 4.3.2、获取请求参数
- 4.3.3、请求乱码问题处理
- 4.3.4、域对象功能
- 4.3.5、请求转发
- 4.4、关于路径写法
- 4.4.1、浏览器路径
- 4.4.2、服务器路径
- 五、登录案例
- 六、`ServletContext`
- 6.1、概述
- 6.2、获取`ServletContext`
- 6.3、作用
- 6.3.1、获取项目在服务器上发布的真实路径
- 6.3.2、获取项目的上下文路径
- 6.3.3、域对象功能
- 6.3.4、打开已经存在的Maven项目_getParameter和getAttribute的区别
一、简介
Server Applet
(服务器小程序),是由服务器端调用和执行的、按照Servlet
自身规范编写的Java
类。
JavaWeb
的三大组件(Servlet
、Filter
、Lisener
)之一,是动态资源的一种。作用:
- 接收请求
- 处理数据
- 完成响应
后续我们学习
Servlet
也是集中在这三点。
二、Servlet
入门案例
2.1、编写Servlet
- 实现
javax.servlet.Servlet
- 重写5个主要方法
- 在核心的
service()
方法中编写输出语句,打印访问结果
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import java.io.IOException;
import java.io.PrintWriter;public class TestServlet implements Servlet {public void init(ServletConfig config) throws ServletException{}public void service(ServletRequest request,ServletResponse response) throws ServletException,IOException{System.out.println("My First Servlet!");}public void destroy(){}public ServletConfig getServletConfig(){return null; } public String getServletInfo(){return null;}
}
2.2、配置Servlet
在
web.xml
中配置Servlet
<?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_3_1.xsd"version="3.1"><!-- 1、添加servlet节点 --><servlet><!-- Servlet的名字,和2中的名字必须一致 --><servlet-name>aServlet</servlet-name><!-- Servlet的全类名 --><servlet-class>tech.code2048.servlet.AServlet</servlet-class></servlet><!-- 2、添加servlet-mapping节点 --><servlet-mapping><!-- Servlet的名字,和1中的名字必须一致 --><servlet-name>aServlet</servlet-name><!-- Servlet的访问路径,通过该路径可以访问doServlet的Get或doPost方法 --><url-pattern>/aServlet</url-pattern></servlet-mapping>
</web-app>
url-pattern
配置的内容就是浏览器地址栏输入的URL
中项目名称后资源的内容
2.3、访问Servlet
在
Tomcat
中部署该项目,访问http://localhost:8080/项目名/testServlet
,控制台及浏览器有输出证明Servlet
部署及访问成功。
2.4、常见错误
500
错误:服务器内部出现了错误修改
TestServlet
代码如下:
package tech.code2048.servlet;import javax.servlet.*;
import java.io.IOException;public class AServlet implements Servlet {@Overridepublic void init(ServletConfig servletConfig) throws ServletException {}@Overridepublic ServletConfig getServletConfig() {return null;}@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {System.out.println("My First Servlet!");String str = null;System.out.println(str.length());}@Overridepublic String getServletInfo() {return null;}@Overridepublic void destroy() {}
}
运行
JavaWeb
项目,访问TestServlet
,页面显示如下:
三、Servlet
详解
Servlet
是单例的,一个类型的Servlet
只有一个实例对象,那么就有可能会出现一个Servlet
同时处理多个请求,Servlet
不是线程安全的,不应该在Servlet
中创建成员变量,因为可能会存在一个线程对这个成员变量进行写操作,另一个线程对这个成员变量进行读操作。
- 不要在
Servlet
中创建成员,创建局部变量即可;- 可以创建无状态成员;
- 可以创建有状态的成员,但状态必须为只读的。
3.1、实现Servlet
的三种方式
最后这个要自己掌握,生命周期的三种方法要背熟!
3.1.1、实现Servlet
接口
javax.servlet.Servlet
代码
public interface Servlet {void init(ServletConfig var1) throws ServletException;ServletConfig getServletConfig();void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;String getServletInfo();void destroy();
}
Servlet
中的大多数方法不由我们来调用,而是由Tomcat
来调用,并且Servlet
的对象也不由我们来创建,由Tomcat
来创建!生命周期方法
init()
- 服务器会在
Servlet
第一次被访问时创建Servlet
,在Servlet
被创建后,服务器会马上调用Servlet
的void init(ServletConfig)
方法;- 在整个
Servlet
的生命周期中,该方法只被调用一次。service()
- 当服务器每次接收到请求时,都会去调用
Servlet
的service()
方法来处理请求;- 每次处理请求都会被调用
destroy()
- 在服务器被关闭时,服务器会去销毁
Servlet
,在销毁Servlet
之前服务器会先去调用Servlet
的destroy()
方法;- 在整个
Servlet
的生命周期中,该方法只被调用一次。生命周期方法方法演示
package tech.code2048.servlet;import javax.servlet.*;
import java.io.IOException;public class BServlet implements Servlet {@Overridepublic void init(ServletConfig servletConfig) throws ServletException {System.out.println("init...");}@Overridepublic ServletConfig getServletConfig() {return null;}@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {System.out.println("service...");}@Overridepublic String getServletInfo() {return null;}@Overridepublic void destroy() {System.out.println("destroy...");}
}
在
web.xml
中配置Servlet
<servlet><servlet-name>bServlet</servlet-name><servlet-class>tech.code2048.servlet.BServlet</servlet-class>
</servlet>
<servlet-mapping><servlet-name>bServlet</servlet-name><url-pattern>/bServlet</url-pattern>
</servlet-mapping>
启动
tomcat
->访问BServlet
->关闭tomcat
观察控制台的打印。
关于
ServletConfig
- 一个
ServletConfig
对象对应一段web.xml
中的配置信息ServletConfig
是一个接口,该接口实现类的对象由Tomcat
提供- 方法
getInitParameter(name)
获取servlet
的初始化参数getInitParameterNames()
getServletContext()
getServletName()
修改
BServlet
代码
package tech.code2048.servlet;import javax.servlet.*;
import java.io.IOException;public class BServlet implements Servlet {@Overridepublic void init(ServletConfig servletConfig) throws ServletException {System.out.println("init...");System.out.println(servletConfig.getInitParameter("name"));}@Overridepublic ServletConfig getServletConfig() {return null;}@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {System.out.println("service...");}@Overridepublic String getServletInfo() {return null;}@Overridepublic void destroy() {System.out.println("destroy...");}
}
修改
web.xml
关于BServlet
的配置
<servlet><servlet-name>bServlet</servlet-name><servlet-class>tech.code2048.servlet.BServlet</servlet-class><!-- BServlet初始化参数 --><init-param><param-name>name</param-name><param-value>zhangsan</param-value></init-param>
</servlet>
<servlet-mapping><servlet-name>bServlet</servlet-name><url-pattern>/bServlet</url-pattern>
</servlet-mapping>
3.1.2、继承GenericServlet
类
GenericServlet
使编写Servlet
变得更容易。它提供生命周期方法init
和destroy
的简单实现,要编写一般的Servlet
,只需重写抽象service
方法即可。代码如下:
package tech.code2048.servlet;import javax.servlet.GenericServlet;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import java.io.IOException;public class CServlet extends GenericServlet {@Overridepublic void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {System.out.println("CServlet....");}
}
web.xml`配置`CServlet
<servlet><servlet-name>cServlet</servlet-name><servlet-class>tech.code2048.servlet.CServlet</servlet-class>
</servlet>
<servlet-mapping><servlet-name>cServlet</servlet-name><url-pattern>/cServlet</url-pattern>
</servlet-mapping>
3.1.3、继承HttpServlet
类
HttpServlet
类是GenericServlet
的子类,它提供了对HTTP
请求的特殊支持,所以通常我们都会通过继承HttpServlet
来完成自定义的Servlet
。在
HttpServlet
的service(HttpServletRequest,HttpServletResponse)
方法会去判断当前请求是GET
还是POST
,如果是GET
请求,那么会去调用本类的doGet()
方法,如果是POST
请求会去调用doPost()
方法,这说明我们在子类中去覆盖doGet()
或doPost()
方法即可。
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import java.io.IOException;public class DServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("doGet.......");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("doPost........");}
}
web.xml`配置`DServlet
<servlet><servlet-name>dServlet</servlet-name><servlet-class>tech.code2048.servlet.DServlet</servlet-class>
</servlet>
<servlet-mapping><servlet-name>dServlet</servlet-name><url-pattern>/dServlet</url-pattern>
</servlet-mapping>
3.2、配置Servlet
的两种方式
3.2.1、web.xml
方式
Servlet2.5
及之前使用该方式
<!-- 1、添加servlet节点 -->
<servlet><!-- Servlet的名字,和2中的名字必须一致 --><servlet-name>aServlet</servlet-name><!-- Servlet的全类名 --><servlet-class>tech.code2048.servlet.AServlet</servlet-class><!-- 配置启动加载 --><load-on-startup>1</load-on-startup>
</servlet>
<!-- 2、添加servlet-mapping节点 -->
<servlet-mapping><!-- Servlet的名字,和1中的名字必须一致 --><servlet-name>aServlet</servlet-name><!-- Servlet的访问路径,通过该路径可以访问doServlet的Get或doPost方法 --><url-pattern>/aServlet</url-pattern>
</servlet-mapping>
url-pattern
定义匹配规则,取值说明:
- 精确匹配
/具体的名称
只有url路径是具体的名称的时候才会触发Servlet
- 后缀匹配
*.xxx
只要是以xxx
结尾的就匹配触发Servlet
- 通配符匹配
/*
匹配所有请求,包含服务器的所有资源- 通配符匹配
/
匹配所有请求,包含服务器的所有资源,不包括jsp
load-on-startup
- 元素标记容器是否应该在
web
应用程序启动的时候就加载这个servlet
;- 它的值必须是一个整数,表示
Servlet
被加载的先后顺序;- 如果该元素的值为负数或者没有设置,则容器会当
Servlet
被请求时再加载;- 如果值为正整数或者0时,表示容器在应用启动时就加载并初始化这个
Servlet
,值越小,Servlet
的优先级越高,就越先被加载。值相同时,容器就会自己选择顺序来加载。
3.2.2、注解方式
Servlet3.0
新增特性,推荐使用
@WebServlet
常用属性:
name
:Servlet
名字,可选value
:配置url
路径,可以配置多个urlPatterns
:配置url
路径 ,和value
作用一样,不能同时和value
使用loadOnStartup
:配置Servlet
的创建的时机, 如果是0或者正数,启动程序时创建,如果是负数,则访问时创建。 数子越小优先级越高。
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;@WebServlet(name = "EServlet", value = "/eServlet", loadOnStartup = 1)
public class EServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("doGet...");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("doPost...");}
}
3.3、常见问题
HTTP Status 404
资源找不到
- 第一种情况:地址书写错误;
- 第二种情况:地址没有问题,把
IDEA
项目中out
目录删除,然后重新运行。
Servlet
地址配置重复,多个Servlet
的url-pattern
使用相同的值
Servlet
地址配置错误,比如没有写/
。
3.4、关于Jrebel
我们在编写
Servlet
时,修改Servlet
或者新增Servlet
都要重启项目让项目重新发布,重启非常耗时,这样会导致效率低下,我们需要一种能够动态发布项目的技术,这就是Jrebel
。
3.4.1、安装
搜索插件
jrebel
, 安装然后重启IDEA
注意:如果使用下面的激活方式,一定要使用
2022.4.1
及之前版本。
3.4.2、激活
- 打开任何一个项目,
Help -> JReBel -> Activation
;- 访问
https://www.guidgen.com
,获取GUID
,然后多次单击Generate new GUID
,复制生成的GUID
;- 在激活页面
Team URL
输入http://jrebel.yanjiayu.cn:9001/2中的GUID
,例如:http://jrebel.yanjiayu.cn:9001/5c556b5a-323f-4803-84e6-6249e7df653f
,直接使用案例的Team URL
也可以;- 邮箱随意填;
- 勾选
I agree....
;- 点击
Activate JRebel
;Next > enable JRebel for projects
;I agree
。
3.4.3、使用之前的配置
File -> Settings -> JRebel&XRebel -> 单击Work offline
;File -> Settings -> Build,Execution,Deployment -> Compiler -> Build project automatically勾上
, 这一步之前我们配置IDEA
的时候配置过;- 展开
IDEA
右侧JRenel
面板,勾选项目后面的第一个多选按钮。注意:我们将来新建的项目只需要配置步骤3就可以了,不需要每次都配置步骤1和步骤2。
3.4.4、启动测试
启动
- 不要选择之前
Tomcat
的启动标志,选择新增的带小火箭的启动标志。测试
- 在已有的
Servlet
中新增内容,新建Servlet
都会立即生效,不需要重启Tomcat
。
四、request
和response
4.1、请求响应流程
4.2、response
对象
reponse
类型为javax.servlet.http.HttpServletResponse
,客户端发出每个请求时,服务器都会创建一个response
对象,并传入给Servlet.service()
方法。response
对象是用来对客户端进行响应的,这说明在service()
方法中使用response
对象可以完成对客户端的响应工作。主要功能
- 设置响应正文
- 设置响应头信息
- 发送状态码
- 重定向
4.2.1、设置响应正文
关于设置响应正文
response
是响应对象,向客户端输出响应正文(响应体)可以使用response
的响应流,常用方法:
PrintWriter out = response.getWriter()
:获取字符流—16ServletOutputStream out = response.getOutputStream()
:获取字节流—8如果响应正文内容为字符,那么使用
response.getWriter()
,如果响应内容是字节,例如下载时,那么可以使用response.getOutputStream()
;注意:在一个请求中,不能同时使用这两个流,也就是说,要么你使用
repsonse.getWriter()
,要么使用response.getOutputStream()
,但不能同时使用这两个流。不然会抛出IllegalStateException
异常。例:在页面打印文字,字体颜色为红色。
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;
import java.io.PrintWriter;@WebServlet(name = "FServlet", value = "/FServlet")
public class FServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {PrintWriter out = response.getWriter();out.println("<p style='color:red;'>Hello World</p>");out.println("<p style='color:red;'>你好世界</p>");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
4.2.2、响应乱码问题处理
问题分析:
- 在使用
response.getWriter()
时默认字符编码为ISO-8859-1
,不支持中文;- 如果希望设置字符流的字符编码为
UTF-8
,可以使用response.setCharacterEncoding(“utf-8”)
来设置,这样可以保证输出给客户端的字符都是使用的UTF-8
编码;- 客户端浏览器并不知道响应数据是什么编码的,浏览器解析之后依然出现乱码。
解决方案:
- 使用
response.setContentType("text/html;charset=utf-8")
;- 一定要在获取输出流前进行设置;
- 优势
- 设置
content-type
响应头,客户端浏览器会使用content-type
头来解读响应数据;- 这个方法还会调用
response.setCharacterEncoding("utf-8")
保证输出给客户端的字符都是使用UTF-8
编码的。上述案例修改
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;
import java.io.PrintWriter;@WebServlet(name = "FServlet", value = "/FServlet")
public class FServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//设置响应头response.setContentType("text/html;charset=utf-8");PrintWriter out = response.getWriter();out.println("<p style='color:red;'>Hello World</p>");out.println("<p style='color:red;'>你好世界</p>");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
4.2.3、重定向
现象:当访问
http://baidu.com
时,发现地址栏变成https://www.baidu.com
,这就是重定向了。概念:重定向是服务器通知浏览器去访问另一个地址,即再发出另一个请求。
实现方式1:
- 设置响应码
response.setStatus(302);
;- 设置重定向的位置
response.setHeader("Location", "/项目名/bServlet");
。实现方式2:
response.sendRedirect("/项目名/bServlet");
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;@WebServlet(name = "GServlet", value = "/GServlet")
public class GServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("GServlet............................");//实现方式1//response.setStatus(302);//response.setHeader("Location", "/JavaWebTest_war_exploded/bServlet");//实现方式2response.sendRedirect("/JavaWebTest_war_exploded/bServlet");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
关于重定向的总结
- 重定向是两次请求
- 重定向的
URL
可以是其他应用,不局限于当前应用,例如重定向到百度- 重定向的响应头为
302
,并且必须要有Location
响应头- 重定向就不要再使用
response.getWriter()
或response.getOutputStream()
输出数据
4.3、request
对象
request
是Servlet.service()
方法的一个参数,类型为javax.servlet.http.HttpServletRequest
。在客户端发出每个请求时,服务器都会创建一个request
对象,并把请求数据封装到request
中,然后在调用Servlet.service()
方法时传递给service()
方法,这说明在service()
方法中可以通过request
对象来获取请求数据。主要功能(平行的关系)
- 获取请求头
- 获取请求参数
- 域对象功能
- 请求转发和请求包含
4.3.1、获取请求头(了解)
String getMethod()`:返回请求方法,例如:`GET
String getRemoteAddr()
:返回当前客户端的IP
地址
String getRemoteHost()
:返回当前客户端的主机名,但这个方法的实现还是获取IP
地址String getServerName()`:返回主机名,例如:`localhost int getServerPort()`:返回服务器端口号,例如:`8080
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;@WebServlet(name = "HServlet", value = "/HServlet")
public class HServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("请求方法:" + request.getMethod());System.out.println("客户端IP:" + request.getRemoteAddr());System.out.println("客户端主机名:" + request.getRemoteHost());System.out.println("主机名:" + request.getServerName());System.out.println("服务器端口号:" + request.getServerPort());System.out.println("客户端端口号:" + request.getRemotePort());}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
4.3.2、获取请求参数
客户端传递参数方式有两种(
GET
和POST
)
- 浏览器地址栏直接输入:一定是
GET
请求- 超链接:一定是
GET
请求- 表单:可以是
GET
,也可以是POST
,这取决与form
的method
属性值
GET
请求和POST
请求的区别
GET
- 可以在请求的URL地址后以
?
的形式带上交给服务器的数据,多个数据之间以&
进行分隔,不安全- 在
URL
地址后附带的参数是有限制的,其数据容量通常不能超过1K
GET
请求没有请求体POST
- 传送的数据量无限制
- 传输的数据在请求体内
- 请求参数不会显示浏览器的地址栏,相对安全
request`获取请求参数的`API
String getParameter(String name)
:通过指定名称获取参数值(必须掌握)String[] getParameterValues(String name)
:当多个参数名称相同时,可以使用方法来获取Enumeration getParameterNames()
:获取所有参数的名字Map getParameterMap()
:获取所有参数封装到Map
中,其中key
为参数名,value
为参数值,因为一个参数名称可能有多个值,所以参数值是String[]
,而不是String
Servlet
代码如下:
package tech.code2048.servlet;import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;
import java.util.Arrays;@WebServlet(name = "IServlet", value = "/IServlet")
public class IServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//获取请求参数String username = request.getParameter("username");System.out.println(username);//当多个参数名称相同时,可以使用方法来获取String[] hobbies = request.getParameterValues("hobby");System.out.println(Arrays.toString(hobbies));}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}
}
Get
方式提交:http://localhost:8080/JavaWebTest_war_exploded/IServlet?username=zs&hobby=睡觉&hobby=game
Post
方式提交:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><form action="/JavaWebTest_war_exploded/IServlet" method="post">用户名<input type="text" name="username" /><br/>爱好<input type="checkbox" name="hobby" value="睡觉1">睡觉1<input type="checkbox" name="hobby" value="睡觉2">睡觉2<input type="checkbox" name="hobby" value="睡觉3">睡觉3<button type="submit">提交</button></form>
</body>
</html>
4.3.3、请求乱码问题处理
Get
请求乱码
Tomcat8
的版本中get
方式不会出现乱码了,因为服务器对url
的编码格式可以进行自动转换。
POST
请求乱码
- 由于客户端是以
UTF-8
字符编码将表单数据传输到服务器端的,因此服务器也需要设置以UTF-8
字符编码进行接收。- 解决方案:使用从
ServletRequest
接口继承而来的setCharacterEncoding(charset)
方法进行统一的编码设置。修改
IServlet
代码如下:
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;
import java.util.Arrays;@WebServlet(name = "IServlet", value = "/IServlet")
public class IServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//设置请求编码request.setCharacterEncoding("utf-8");//获取请求参数String username = request.getParameter("username");System.out.println(username);//当多个参数名称相同时,可以使用方法来获取String[] hobbies = request.getParameterValues("hobby");System.out.println(Arrays.toString(hobbies));}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}
}
4.3.4、域对象功能
- 域对象作用就是在多个
Servlet
之间传递数据request
可以在一个请求中共享数据,一个请求会创建一个request
对象,如果在一个请求中经历了多个Servlet
,那么多个Servlet
就可以使用request
来共享数据- 域方法
- 存
void setAttribute(String name, Object value)
- 取
Object getAttribute(String name)
- 删除
void removeAttribute(String name)
- 一个请求可以通过转发、包含跨越多个
Servlet
,在这个请求链之上的Servlet
可以通过request
共享数据
4.3.5、请求转发
转发的作用在服务器端,将请求发送给服务器上的其他资源,以共同完成一次请求的处理,多个
Servlet
或JSP
共同来处理一个请求。
实现步骤:
- 创建调度器
RequestDispatcher rd = request.getRequestDispatcher("/BServlet");
- 转发
rd.forward(request, response);
以下代码实现转发,并通过request域传递数据
@WebServlet(name = "JServlet", value = "/JServlet")
public class JServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//在request域中存放数据request.setAttribute("name", "zs");//转发//创建调度器RequestDispatcher dispatcher = request.getRequestDispatcher("/KServlet");//转发dispatcher.forward(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}
}@WebServlet(name = "KServlet", value = "/KServlet")
public class KServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//从request域中获取数据String name = (String)request.getAttribute("name");System.out.println(name);//从request域中删除数据request.removeAttribute("name");//转发request.getRequestDispatcher("/LServlet").forward(request, response);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}
}@WebServlet(name = "LServlet", value = "/LServlet")
public class LServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//从request域中获取数据String name = (String)request.getAttribute("name");//由于KServlet删除了域对象中的内容,导致name输出为nullSystem.out.println("LServlet:" + name);}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
请求转发和重定向的区别
- 请求转发是一个请求,而重定向是两个请求
- 请求转发后浏览器地址栏不会有变化,而重定向会有变化,因为重定向是两个请求
- 请求转发的目标只能是本应用中的资源,重定向的目标可以是其他应用
- 请求转发对
AServlet
和BServlet
的请求方法是相同的,即要么都是GET
,要么都是POST
,因为请求转发是一个请求- 重定向的第二个请求一定是
GET
4.4、关于路径写法
4.4.1、浏览器路径
超链接、表单、重定向都是客户端路径(假设当前主机是
http://localhost:8080
,应用名称Test)
绝对路径
http://www.baidu.com
相对路径
以
/
开头的相对路径
- 表示相对于当前主机地址(
http://localhost:8080
)- 在超链接、表单、重定向,如果以
/
开头,一定是/Test/...
- 参考重定向中路径的写法
不以
/
开头的相对路径
- 表示相对当前路径
注意:强烈建议使用
/
开头的相对路径,后面是当前应用的名称,再是访问路径。
4.4.2、服务器路径
这里说的服务器路径,就是以
/
开头的相对路径
- 表示相对于当前应用(
http://localhost:8080/Test
)- 参考请求转发中路径的写法
五、登录案例
使用
Servlet
完成登录功能,不需要连接数据库,需求如下:
- 输入用户名:
admin
,密码:admin
,页面显示登录成功
;- 输入其他用户名密码,页面显示
登录失败
。登录页代码如下:
<!DOCTYPE html>
<html><head><meta charset="utf-8" /><title>登录</title></head><body><form action="http://localhost:8080/JavaWebTest/LoginServlet" method="post"><fieldset style="width: 300px;"><legend>用户登录</legend><p><label>账号</label><input type="text" name="username" placeholder="请输入用户名" /></p><p><label>密码</label><input type="password" name="password" placeholder="请输入密码" /></p><p><button type="submit">登录</button><button type="reset">重置</button></p></fieldset></form></body>
</html>
登录成功页
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>success</title>
</head>
<body><p>登录成功</p>
</body>
</html>
登录失败页
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>error</title>
</head>
<body><p>登录失败</p>
</body>
</html>
Servlet代码
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;@WebServlet(name = "LoginServlet", value = "/LoginServlet")
public class LoginServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//获取请求参数String username = request.getParameter("username");String password = request.getParameter("password");if(username.equals("admin") && password.equals("admin")) {request.getRequestDispatcher("/success.html").forward(request, response);} else {request.getRequestDispatcher("/error.html").forward(request, response);}}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}
}
六、ServletContext
6.1、概述
一个项目只有一个
ServletContext
对象,使用它可以给多个Servlet
传递数据,与天地同寿,这个对象在Tomcat
启动时就创建,在Tomcat
关闭时才会消失。特点:
- 唯一性:一个应用对应一个
ServletContext
。- 生命周期:只要容器不关闭或者应用不卸载,
ServletContext
就一直存在。
6.2、获取ServletContext
GenericServlet
提供了getServletContext()
方法;HttpServletRequest
提供了getServletContext()
方法;HttpSession
提供了getServletContext()
方法。
6.3、作用
6.3.1、获取项目在服务器上发布的真实路径
//获取项目在服务器上的真实路径
String realpath=servletContext.getRealPath("/");
6.3.2、获取项目的上下文路径
上下文路径指的是应用名称。
//获取项目的上下文路径 - 上下文路径(应用程序名称)
System.out.println(servletContext.getContextPath());
//另一种方式
System.out.println(request.getContextPath());
案例
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;@WebServlet(name = "MServlet", value = "/MServlet")
public class MServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {//获取servletContextServletContext servletContext = this.getServletContext();//获取项目真实路径System.out.println(servletContext.getRealPath("/"));//获取项目上下文路径System.out.println(servletContext.getContextPath());System.out.println(request.getContextPath());}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
6.3.3、域对象功能
ServletContext
拥有作用域,可以存储数据到全局容器中
- 存储数据:
servletContext.setAttribute("name",value);
- 获取数据:
servletContext.getAttribute("name");
- 移除数据:
servletContext.removeAttribute("name");
案例,统计网站访问人数
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;@WebServlet(name = "NServlet", value = "/NServlet")
public class NServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {ServletContext sctx = this.getServletContext();Integer count = (Integer) sctx.getAttribute("count");if(count == null) {count = 1; //自动装箱} else {count++;}sctx.setAttribute("count", count);PrintWriter out = response.getWriter();out.println("<h1>" + count + "</h1>");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {}
}
6.3.4、打开已经存在的Maven项目_getParameter和getAttribute的区别
1.发给别人的项目里面只需要有下面两个东西就行了,别的东西都不要(尤其是.idea和.ml,这都是自己本地配置的东西,不需要发给别人)
2.关于打开项目:
打开项目,可以选择到pom.xml打开,信任项目,看看里面文件夹的颜色是否正确。
3.自己配置项目
(tomcat…跟之前配置方法一样)
4.getParameter和getAttribute的区别:
问获取请求参数getParameter和域对象什么关系?
答:没有任何关系,是完全平行的两个功能
那什么时候需要获取请求参数getParameter?什么时候才要从域对象里面获取东西?
答:如果网页、浏览器给服务器发送数据,要获取浏览器发送过来的数据,用第一个获取请求参数getParameter。
如果请求域中存放了数据,想在转发过程中想获取数据,也就是在Servlet间分享数据的时候获取数据,那么用第二个从域对象里面获取东西。
5.在同一个域当中(在同一个ServletContext中),如果同一个属性多次赋值,那么相同的key,新的值会覆盖(替换)旧的值