尚硅谷javaweb视频
JavaWeb的内容不用全看,重点看Servlet、Filter、Listener这三大组件,了解Session、cookie的使用;
文章目录
一、JavaWeb 的概念
1、什么是 JavaWeb2、什么是请求3、什么是响应4、请求和响应的关系5、Web 资源的分类6、常用的 Web 服务器 二、Tomcat
1.Tomcat的使用
(1)安装(2)目录介绍(3)如何启动 Tomcat 服务器(4)Tomcat 的停止(5)如何修改 Tomcat 的端口号(6)如何部暑 web 工程到 Tomcat 中(7)手托 html 页面到浏览器和在浏览器中输入 http://ip:端口号/工程名/访问的区别(8)ROOT 的工程的访问,以及 默认 index.html 页面的访问 2、IDEA 整合 Tomcat 服务器,创建web项目 三、Servlet技术
1、什么是Servlet2、第一个Servlet程序
(1) url地址到Servlet程序的访问(2)Servlet 的生命周期(3)GET 和 POST 请求的分发处理(4)Servlet 类的继承体系 3.ServletConfig 类
(1)ServletConfig 类的三大作用 4.ServletContext 类
(1)什么是 ServletContext?(2)ServletContext 类的四个作用 5.HTTP 协议
(1)什么是 HTTP 协议(2)请求的 HTTP 协议格式(3)响应的 HTTP 协议格式(4)常用的响应码说明(5)MIME 类型说明 6.HttpServletRequest 类
(1)HttpServletRequest 类有什么作用。(2)HttpServletRequest 类的常用方法(3)如何获取请求参数(4)请求的转发(5) base 标签的作用(6)Web 中的相对路径和绝对路径(7)web 中 / 斜杠的不同意义 7.HttpServletResponse 类
(1)HttpServletResponse 类的作用(2)两个输出流的说明。(3)如何往客户端回传数据(4)响应的乱码解决(5)请求重定向 四、cookie
1、什么是 cookie?2、如何创建 cookie3、服务器如何获取 cookie4、cookie 值的修改5、cookie 生命控制6、cookie 有效路径 Path 的设置7、cookie 练习---免输入用户名登录 五、Session
1、什么是 Session 会话?2、如何创建 Session 和获取(id 号,是否为新)3、Session 域数据的存取4、Session 生命周期控制5、浏览器和 Session 之间关联的技术内幕 六、Filter过滤器
1、Filter 什么是过滤器2、Filter 的初体验3、Filter 的生命周期4、FilterConfig 类5、FilterChain 过滤器链6、Filter 的拦截路径 七、监听器
1、监听器的概述
(1)什么是监听器(2)监听器的用途(3)监听器的术语 2、监听器入门3、Servlet中的监听器
(1)简介(2)监听器的分类(一)监听三个域对象的创建和销毁的监听器(三个)
(1)ServletContextListener监听器
①ServletContextListener监听器的作用②ServletContext创建和销毁③ServletContext创建和销毁④ServletContextListener企业用途 (2)HttpSessionListener监听器
①HttpSessionListener监听器的作用②HttpSessionListener创建和销毁③编写监听器监听HttpSession对象创建和销毁 (3)ServletRequestListener监听器
①ServletRequestListener监听器的作用②ServletRequestListener创建和销毁③编写监听器监听ServletRequest对象创建和销毁 (4)统计当前在线人数 (二) 监听三个域对象的属性变更(属性添加、移除、替换)的监听器(三个)
(1)ServletContextAttributeListener(2)HttpSessionAttributeListener(3)ServletRequestAttributeListener(4)以HttpSessionAttributeListener为例 (三) 监听HttpSession中JavaBean的状态改变(钝化、活化、绑定、解除绑定)的监听(两个)
(1)HttpSessionBindingListener监听器(2)HttpSessionActivationListener监听器
一、JavaWeb 的概念 1、什么是 JavaWeb
JavaWeb 是指,所有通过 Java 语言编写可以通过浏览器访问的程序的总称,叫 JavaWeb。
JavaWeb 是基于请求和响应来开发的。
请求是指客户端给服务器发送数据,叫请求 Request。
3、什么是响应响应是指服务器给客户端回传数据,叫响应 Response。
4、请求和响应的关系请求和响应是成对出现的,有请求就有响应。
5、Web 资源的分类web 资源按实现的技术和呈现的效果的不同,又分为静态资源和动态资源两种。
6、常用的 Web 服务器静态资源: html、css、js、txt、mp4 视频 , jpg 图片
动态资源: jsp
Tomcat:由 Apache 组织提供的一种 Web 服务器,提供对 jsp 和 Servlet 的支持。它是一种轻量级的 javaWeb 容器(服务器),也是当前应用最广的 JavaWeb 服务器(免费)。Jboss:是一个遵从 JavaEE 规范的、开放源代码的、纯 Java 的 EJB 服务器,它支持所有的 JavaEE 规范(免费)。GlassFish: 由 Oracle 公司开发的一款 JavaWeb 服务器,是一款强健的商业服务器,达到产品级质量(应用很少)。Resin:是 CAUCHO 公司的产品,是一个非常流行的服务器,对 servlet 和 JSP 提供了良好的支持,性能也比较优良,resin 自身采用 JAVA 语言开发(收费,应用比较多)。WebLogic:是 Oracle 公司的产品,是目前应用最广泛的 Web 服务器,支持 JavaEE 规范,而且不断的完善以适应新的开发要求,适合大型项目(收费,用的不多,适合大公司)。
二、Tomcat 1.Tomcat的使用 (1)安装
找到你需要用的 Tomcat 版本对应的 zip 压缩包,解压到需要安装的目录即可。
(2)目录介绍bin 专门用来存放 Tomcat 服务器的可执行程序conf 专门用来存放 Tocmat 服务器的配置文件lib 专门用来存放 Tomcat 服务器的 jar 包logs 专门用来存放 Tomcat 服务器运行时输出的日记信息temp 专门用来存放 Tomcdat 运行时产生的临时数据webapps 专门用来存放部署的 Web 工程。work 是 Tomcat 工作时的目录,用来存放 Tomcat 运行时 jsp 翻译为 Servlet 的源码,和 Session 钝化的目录。 (3)如何启动 Tomcat 服务器
找到 Tomcat 目录下的 bin 目录下的 startup.bat 文件,双击,就可以启动 Tomcat 服务器。
如何测试 Tomcat 服务器启动成功???
打开浏览器,在浏览器地址栏中输入以下地址测试:
1、http://localhost:8080
2、http://127.0.0.1:8080
3、http://真实 ip:8080
当出现如下界面,说明 Tomcat 服务器启动成功!!!
另一种启动 tomcat 服务器的方式
1、打开命令行
2、cd 到 你的 Tomcat 的 bin 目录下
1、点击 tomcat 服务器窗口的 x 关闭按钮
2、把 Tomcat 服务器窗口置为当前窗口,然后按快捷键 Ctrl+C
3、找到 Tomcat 的 bin 目录下的 shutdown.bat 双击,就可以停止 Tomcat 服务器
Mysql 默认的端口号是:3306
Tomcat 默认的端口号是:8080
找到 Tomcat 目录下的 conf 目录,找到 server.xml 配置文件
第一种部署方法:只需要把 web 工程的目录拷贝到 Tomcat 的 webapps 目录下
即可。
1、在 webapps 目录下创建一个 book 工程
2、把书城第一阶段的内容拷贝到里面
3、如何访问 Tomcat 下的 web 工程。
只需要在浏览器中输入访问地址格式如下:
(7)手托 html 页面到浏览器和在浏览器中输入 http://ip:端口号/工程名/访问的区别 (8)ROOT 的工程的访问,以及 默认 index.html 页面的访问 2、IDEA 整合 Tomcat 服务器,创建web项目http://ip:port/工程名/目录下/文件名
三、Servlet技术 1、什么是Servlet
- Servlet 是 JavaEE 规范之一。规范就是接口Servlet 就 JavaWeb 三大组件之一。三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。Servlet 是运行在服务器上的一个 java 小程序,它可以接收客户端发送过来的请求,并响应数据给客户端。
1、编写一个类去实现 Servlet 接口
2、实现 service 方法,处理请求,并响应数据
3、到 web.xml 中去配置 servlet 程序的访问地址
Servlet 程序的示例代码:
1、编写一个类去实现 Servlet 接口
2、实现 service 方法,处理请求,并响应数据
3、到 web.xml 中去配置 servlet 程序的访问地址
1、执行 Servlet 构造器方法
2、执行 init 初始化方法
第一、二步,是在第一次访问,的时候创建 Servlet 程序会调用。
3、执行 service 方法
第三步,每次访问都会调用。(不断刷新不断访问 )
4、执行 destroy 销毁方法
(3)GET 和 POST 请求的分发处理第四步,在 web 工程停止的时候调用。
package com.cen.servlet;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
public class TestServlet implements Servlet {
public TestServlet() {
System.out.println("1、构造器方法");
}
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("2、init初始化方法");
}
@Override
public ServletConfig getServletConfig() {
return null;
}
//Service方法用来处理请求和响应
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("3、hello被访问了");
//区分post和get方式
HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
String method = httpServletRequest.getMethod();
if ("GET".equals(method)){
doGet();
}else if ("POST".equals(method)){
doPost();
}
}
//做get的请求
public void doGet(){
System.out.println("get请求");
}
//做get的请求
public void doPost(){
System.out.println("post请求");
}
@Override
public String getServletInfo() {
return null;
}
@Override
public void destroy() {
System.out.println("4、destory销毁方法");
}
}
(4)Servlet 类的继承体系
3.ServletConfig 类
ServletConfig 类从类名上来看,就知道是 Servlet 程序的配置信息类。
Servlet 程序和 ServletConfig 对象都是由 Tomcat 负责创建,我们负责使用。
Servlet 程序默认是第一次访问的时候创建,ServletConfig 是每个 Servlet 程序创建时,就创建一个对应的 ServletConfig 对象。
1、可以获取 Servlet 程序的别名 servlet-name 的值
2、获取初始化参数 init-param
3、获取 ServletContext 对象
web.xml 中的配置:
HelloServlet com.atguigu.servlet.HelloServlet username root url jdbc:mysql://localhost:3306/test HelloServlet /hello
Servlet 中的代码:
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("2 init 初始化方法");
// 1、可以获取 Servlet 程序的别名 servlet-name 的值
System.out.println("HelloServlet 程序的别名是:" + servletConfig.getServletName());
// 2、获取初始化参数 init-param
System.out.println("初始化参数 username 的值是;" + servletConfig.getInitParameter("username"));
System.out.println("初始化参数 url 的值是;" + servletConfig.getInitParameter("url"));
// 3、获取 ServletContext 对象
System.out.println(servletConfig.getServletContext());
}
注意:
1、ServletContext 是一个接口,它表示 Servlet 上下文对象
2、一个 web 工程,只有一个 ServletContext 对象实例。
3、ServletContext 对象是一个域对象。
4、ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候销毁。
即例如(2)的演示
什么是域对象?
域对象,是可以像 Map 一样存取数据的对象,叫域对象。
这里的域指的是存取数据的操作范围,整个 web 工程。
存数据 取数据 删除数据 Map put() get() remove() 域对象 setAttribute() getAttribute() removeAttribute()(2)ServletContext 类的四个作用
1、获取 web.xml 中配置的上下文参数 context-param
2、获取当前的工程路径,格式: /工程路径
3、获取工程部署后在服务器硬盘上的绝对路径
4、像 Map 一样存取数据
ServletContext 演示代码:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1、获取 web.xml 中配置的上下文参数 context-param
ServletContext context = getServletConfig().getServletContext();
String username = context.getInitParameter("username");
System.out.println("context-param 参数 username 的值是:" + username);
System.out.println("context-param 参数 password 的值是:" +context.getInitParameter("password"));
// 2、获取当前的工程路径,格式: /工程路径
System.out.println( "当前工程路径:" + context.getContextPath() );
// 3、获取工程部署后在服务器硬盘上的绝对路径
System.out.println("工程部署的路径是:" + context.getRealPath("/"));
System.out.println("工程下 css 目录的绝对路径是:" + context.getRealPath("/css"));
System.out.println("工程下 imgs 目录 1.jpg 的绝对路径是:" + context.getRealPath("/imgs/1.jpg"));
}
web.xml 中的配置:
username context password root
ServletContext 像 Map 一样存取数据:
ContextServlet1 代码:
public class TestServlet5 extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取ServletContext对象
ServletContext context = getServletContext();
System.out.println("保存之前: Context1 获取 key1 的值是:"+ context.getAttribute("key1"));
//重启时值为空,若执行了TestContext1之后,因为执行了往里边存东西的操作
// 由于ServletContext 是在 web 工程部署启动的时候创建。在 web 工程停止的时候才会销毁。它的值将不会为空
context.setAttribute("key1","value1");
System.out.println("Context1 中获取域数据key1的值是:"+context.getAttribute("key1"));
}
}
ContextServlet2 代码:
public class TestServlet55 extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
ServletContext context = getServletContext();
System.out.println("保存之前: Context1 获取 key1 的值是:"+ context.getAttribute("key1"));
System.out.println("Context2 中获取域数据key1的值是:"+context.getAttribute("key1"));
}
}
5.HTTP 协议
(1)什么是 HTTP 协议
什么是协议?
协议是指双方,或多方,相互约定好,大家都需要遵守的规则,叫协议。
所谓 HTTP 协议,就是指,客户端和服务器之间通信时,发送的数据,需要遵守的规则,叫 HTTP 协议。
HTTP 协议中的数据又叫报文。
客户端给服务器发送数据叫请求。
服务器给客户端回传数据叫响应。
请求又分为 GET 请求,和 POST 请求两种
i. GET 请求
1、请求行
(1) 请求的方式 GET
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号 HTTP/1.1
2、请求头 key :
value 组成 不同的键值对,表示不同的含义。
空行
3、请求体 ===>>> 就是发送给服务器的数据
ii. POST 请求
1、请求行
(1) 请求的方式 POST
(2) 请求的资源路径[+?+请求参数]
(3) 请求的协议的版本号HTTP/1.1
2、请求头
(1)key : value 不同的请求头,有不同的含义
空行
3、请求体 ===>>>就是发送给服务器的数据
iii. 常用请求头的说明
Accept: 表示客户端可以接收的数据类型
Accpet-Languege: 表示客户端可以接收的语言类型
User-Agent: 表示客户端浏览器的信息
Host: 表示请求时的服务器 ip 和端口号
(3)响应的 HTTP 协议格式iv. 哪些是 GET 请求,哪些是 POST 请求
GET 请求有哪些:
1、form 标签 method=get
2、a 标签
3、link 标签引入 css
4、script 标签引入 js 文件
5、img 标签引入图片
6、iframe 引入 html 页面
7、在浏览器地址栏中输入地址后敲回车POST 请求有哪些:
8、form 标签 method=post
1、响应行
(1) 响应的协议和版本号
(2) 响应状态码
(3) 响应状态描述符
2、响应头
(1) key : value 不同的响应头,有其不同含义
3、响应体 ---->>> 就是回传给客户端的数据
200 表示请求成功
302 表示请求重定向(明天讲)
404 表示请求服务器已经收到了,但是你要的数据不存在(请求地址错误)
500 表示服务器已经收到请求,但是服务器内部错误(代码错误)
MIME 是 HTTP 协议中数据类型。
MIME 的英文全称是"Multipurpose Internet Mail Extensions" 多功能 Internet 邮件扩充服务。MIME 类型的格式是“大类型/小类型”,并与某一种文件的扩展名相对应。
常见的 MIME 类型:
每次只要有请求进入 Tomcat 服务器,Tomcat 服务器就会把请求过来的 HTTP 协议信息解析好封装到 Request 对象中。
然后传递到 service 方法(doGet 和 doPost)中给我们使用。我们可以通过 HttpServletRequest 对象,获取到所有请求的
信息。
getRequestURI() 获取请求的资源路径getRequestURL() 获取请求的统一资源定位符(绝对路径)getRemoteHost() 获取客户端的 ip 地址getHeader() 获取请求头getParameter() 获取请求的参数getParameterValues() 获取请求的参数(多个值的时候使用)getMethod() 获取请求的方式 GET 或 POSTsetAttribute(key, value); 设置域数据getAttribute(key); 获取域数据getRequestDispatcher() 获取请求转发对象
常用 API 示例代码:
public class RequestAPIServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//i. getRequestURI() 获取请求的资源路径
System.out.println("URL="+req.getRequestURI());//URL=/2_Servlet/requestAPIServlet
//ii. getRequestURL() 获取请求的统一资源定位符(绝对路径)
System.out.println("URL="+req.getRequestURL());//URL=http://localhost:8088/2_Servlet/requestAPIServlet
//iii. getRemoteHost() 获取客户端的 ip 地址
System.out.println("客户端ip地址:"+req.getRemoteHost());
//iv. getHeader() 获取请求头
System.out.println("URL="+req.getHeader("User-Agent"));
//vii. getMethod() 获取请求的方式 GET 或 POST
System.out.println("请求的方式"+req.getMethod());
}
}
(3)如何获取请求参数
表单:
Java 代码:
public class ParameterServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//设置请求体的字符集为UTF-8,从而解决post请求的中文问题,与get请求不同,post请求不设置会乱码
//该API也要在获取请求参数之前才能调用
req.setCharacterEncoding("UTF-8");
System.out.println("------doget--------");
//获取请求参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String hobby = req.getParameter("hobby");
System.out.println("用户名:"+username);
System.out.println("密码:"+password);
System.out.println("兴趣:"+hobby);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("------dopost--------");
//获取请求参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
System.out.println("用户名:"+username);
System.out.println("密码:"+password);
//数组可以放多个值
System.out.println("兴趣爱好:" + Arrays.asList(hobby))
}
doGet 请求的中文乱码解决:
// 获取请求参数
String username = req.getParameter("username");
//1 先以 iso8859-1 进行编码
//2 再以 utf-8 进行解码
username = new String(username.getBytes("iso-8859-1"), "UTF-8");
POST 请求的中文乱码解决
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException,IOException {
// 设置请求体的字符集为 UTF-8,从而解决 post 请求的中文乱码问题
req.setCharacterEncoding("UTF-8");
System.out.println("-------------doPost------------");
// 获取请求参数
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobby = req.getParameterValues("hobby");
System.out.println("用户名:" + username);
System.out.println("密码:" + password);
System.out.println("兴趣爱好:" + Arrays.asList(hobby))
(4)请求的转发
什么是请求的转发?
请求转发是指,服务器收到请求后,从一次资源跳转到另一个资源的操作叫请求转发。
Servlet1代码
public class Servlet1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet1(柜台1)中查看参数(材料):"+username);
//给材料 盖一个章,并传递到Servlet2(柜台2)去查看
req.setAttribute("key1","柜台1的章");
//问题:Servlet2(柜台2怎么走)
RequestDispatcher requestDispatcher = req.getRequestDispatcher("/Servlet2");
//RequestDispatcher requestDispatcher = req.getRequestDispatcher("/WEB-INF/form.html");//则可以通过请求转发访问该目录
//走向Servlet2(柜台2)
requestDispatcher.forward(req,resp);
}
}
Servlet2代码
public class Servlet2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获取请求的参数(办事材料)查看
String username = req.getParameter("username");
System.out.println("在Servlet2(柜台2)中查看参数(材料):"+username);
//查看是否有柜台1的章
Object key1 = req.getAttribute("key1");
System.out.println("柜台1是否有章:"+key1);
//处理自己到业务
System.out.println("Servlet2 处理自己到业务");
}
}
结果
html页面
Title
这是web下的index.html页面
c.html
请求转发c.html
Title
这是a下的b下的c.html页面
跳回首页
public class ForWardC extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("经过了ForwardC程序");
req.getRequestDispatcher("/a.b/c.html").forward(req, resp);
}
}
(6)Web 中的相对路径和绝对路径
在 javaWeb 中,路径分为相对路径和绝对路径两种:
相对路径是:
. 表示当前目录
… 表示上一级目录
资源名 表示当前目录/资源名
绝对路径:
http://ip:port/工程路径/资源路径
在实际开发中,路径都使用绝对路径,而不简单的使用相对路径。
1、绝对路径
2、base+相对
在 web 中 / 斜杠 是一种绝对路径。
/ 斜杠 如果被浏览器解析,得到的地址是:http://ip:port/
< a href="/">斜杠< /a >
/ 斜杠 如果被服务器解析,得到的地址是:http://ip:port/工程路径
使用场景: 1、7.HttpServletResponse 类 (1)HttpServletResponse 类的作用/servlet1 2、servletContext.getRealPath(“/”); 3、request.getRequestDispatcher(“/”); 特殊情况: response.sendRediect(“/”); 把斜杠发送给浏览器解析。得到 http://ip:port/
HttpServletResponse 类和 HttpServletRequest 类一样。每次请求进来,Tomcat 服务器都会创建一个 Response 对象传递给 Servlet 程序去使用。
HttpServletRequest 表示请求过来的信息,HttpServletResponse 表示所有响应的信息,我们如果需要设置返回给客户端的信息,都可以通过HttpServletResponse 对象来进行设置
(2)两个输出流的说明。字节流 getOutputStream(); 常用于下载(传递二进制数据)
字符流 getWriter(); 常用于回传字符串(常用)
两个流同时只能使用一个。
使用了字节流,就不能再使用字符流,反之亦然,否则就会报错。
要求 : 往客户端回传 字符串 数据。
public class ResponseIOServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException,IOException {
// 要求 : 往客户端回传 字符串 数据。
PrintWriter writer = resp.getWriter();
writer.write("response's content!!!");
}
}
(4)响应的乱码解决
解决响应中文乱码方案一(不推荐使用):
// 设置服务器字符集为 UTF-8
resp.setCharacterEncoding("UTF-8");
// 通过响应头,设置浏览器也使用 UTF-8 字符集
resp.setHeader("Content-Type", "text/html; charset=UTF-8");
解决响应中文乱码方案二(推荐):
// 它会同时设置服务器和客户端都使用 UTF-8 字符集,还设置了响应头
// 此方法一定要在获取流对象之前调用才有效
resp.setContentType("text/html; charset=UTF-8");
(5)请求重定向
请求重定向,是指客户端给服务器发请求,然后服务器告诉客户端说。我给你一些地址。你去新地址访问。叫请求重定向(因为之前的地址可能已经被废弃)。
请求重定向的第一种方案:
// 设置响应状态码 302 ,表示重定向,(已搬迁)
resp.setStatus(302);
// 设置响应头,说明 新的地址在哪里
resp.setHeader("Location", "http://localhost:8080");
请求重定向的第二种方案(推荐使用):
resp.sendRedirect("http://localhost:8080");
Response1
public class Response1 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("曾到此一游 Response1 ");
// req.setAttribute("key1", "value1");
//第一种
// 设置响应状态码302 ,表示重定向,(已搬迁)
resp.setStatus(302);
// 设置响应头,说明 新的地址在哪里
resp.setHeader("Location", "http://localhost:8088/2_Servlet/response2");
// resp.setHeader("Location", "http://localhost:8080");
//第二种
// resp.sendRedirect("http://localhost:8088");
}
}
Response2
public class Response2 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(req.getAttribute("key1"));
resp.getWriter().write("response2's result!");
}
}
即访问Response1的时候显示Response2的结果
四、cookie 1、什么是 cookie?- cookie 翻译过来是饼干的意思。cookie 是服务器通知客户端保存键值对的一种技术。客户端有了 cookie 后,每次请求都发送给服务器。每个 cookie 的大小不能超过 4kb
Servlet 程序中的代码:
//创建cookie
protected void createcookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.创建cookie对象
cookie cookie = new cookie("key1","value1");
cookie cookie1 = new cookie("key2","value2");
cookie cookie2 = new cookie("key3","value3");
//2.通知客户端保存cookie
resp.addcookie(cookie);
resp.addcookie(cookie1);
resp.addcookie(cookie2);
resp.getWriter().write("cookie创建成功");
}
3、服务器如何获取 cookie
服务器获取客户端的 cookie 只需要一行代码:req.getcookies():cookie[]
cookie 的工具类:
public class cookieUtils {
public static cookie findcookie(String name,cookie[] cookies){
if (name == null || cookies == null || cookies.length == 0){
return null;
}
for (cookie cookie : cookies){
if (name.equals(cookie.getName())){
return cookie;
}
}
return null;
}
}
Servlet 程序中的代码:
//获取cookie
protected void getcookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
cookie[] cookies = req.getcookies();
for (cookie cookie:cookies){
//getName方法返回cookie的key(名)
//getValue方法返回cookie的value值
resp.getWriter().write("cookie["+cookie.getName()+"="+cookie.getValue()+"]
");
}
cookie iWantcookie = cookieUtils.findcookie("key2",cookies);
// for (cookie cookie:cookies){
// if ("key1".equals(cookie.getName())){
// iWantcookie = cookie;
// break;
// }
// }
//如果不等于null,说明赋过值,也就是找到了相应的cookie值
if (iWantcookie != null){
resp.getWriter().write("找到了需要的cookie");
}
}
4、cookie 值的修改
方案一:
1、先创建一个要修改的同名(指的就是 key)的 cookie 对象
2、在构造器,同时赋于新的 cookie 值。
3、调用 response.addcookie( cookie );通知 客户端 保存修改
方案二:
1、先查找到需要修改的 cookie 对象
2、调用 setValue()方法赋于新的 cookie 值。
3、调用 response.addcookie()通知客户端保存修改
//cookie的修改
protected void updatecookie(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 方案一:
// 1、先创建一个要修改的同名(指的就是 key)的 cookie 对象
// 2、在构造器,同时赋于新的 cookie 值。
// cookie cookie = new cookie("key1","newValue1");
// 3、调用 response.addcookie(cookie);
// resp.addcookie(cookie);
// resp.getWriter().write("key1的cookie已经修改完成");
// 方案二:
// 1、先查找到需要修改的 cookie 对象
cookie cookie = cookieUtils.findcookie("key2",req.getcookies());
if (cookie != null){
// 2、调用 setValue()方法赋于新的 cookie 值。
cookie.setValue("newValue?");
// 3、调用 response.addcookie()通知客户端保存修改
resp.addcookie(cookie);
}
resp.getWriter().write("key1的cookie已经修改好");
}
5、cookie 生命控制
cookie 的生命控制指的是如何管理 cookie 什么时候被销毁(删除)
setMaxAge()
正数,表示在指定的秒数后过期
负数,表示浏览器一关,cookie 就会被删除(默认值是-1)
零,表示马上删除 cookie
//浏览器的存活设置
protected void default3600(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
cookie cookie = new cookie("life3600","life3600");
cookie.setMaxAge(60*60);//设置cookie一小时后被删除。无效
resp.addcookie(cookie);
resp.getWriter().write("已经设置可存活一小时的cookie");
}
protected void defaultnow(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//先找到要删除的cookie对象
cookie cookie = cookieUtils.findcookie("key3",req.getcookies());
if (cookie != null){
//调用setMaxAge(0);
cookie.setMaxAge(0);//表示马上删除,都不需要等待浏览器关闭
//调用response.addcookie(cookie);
resp.addcookie(cookie);
resp.getWriter().write("key3的cookie已经被删除");
}
}
protected void defaultLife(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
cookie cookie = new cookie("defaultLife","defaultLife");
cookie.setMaxAge(-1);//设置存活时间,但实际上默认值就为-1
resp.addcookie(cookie);
}
6、cookie 有效路径 Path 的设置
cookie 的 path 属性可以有效的过滤哪些 cookie 可以发送给服务器。哪些不发。
path 属性是通过请求的地址来进行有效的过滤。
cookieA path=/工程路径
cookieB path=/工程路径/abc
请求地址如下:
http://ip:port/工程路径/a.html
cookieA 发送
cookieB 不发送
http://ip:port/工程路径/abc/a.html
cookieA 发送
cookieB 发送
//设置path路径
protected void testPath(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
cookie cookie = new cookie("path1","path1");
//getContextPath() ===>>>得到工程路径
cookie.setPath(req.getContextPath()+"/abc");//即满足abc路径条件的基础上才给看见cookie,起到了一个过滤的作用
resp.addcookie(cookie);
resp.getWriter().write("创建了一个带有Path路径的cookie ");
}
7、cookie 练习—免输入用户名登录
login.jsp 页面
LoginServlet 程序:
public class LoginServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String username = req.getParameter("username");
String password = req.getParameter("password");
if ("cencen".equals(username) && "123456".equals(password)){
//登陆成功
cookie cookie = new cookie("username",username);
cookie.setMaxAge(60*60*24*7);//当前cookie一周内有效
resp.addcookie(cookie);
System.out.println("登录成功");
}else {
//登录失败
System.out.println("登录失败");
}
}
}
五、Session
1、什么是 Session 会话?
1、Session 就一个接口(HttpSession)。
2、Session 就是会话。它是用来维护一个客户端和服务器之间关联的一种技术。
3、每个客户端都有自己的一个 Session 会话。
4、Session 会话中,我们经常用来保存用户登录之后的信息。
如何创建和获取 Session。它们的 API 是一样的。
request.getSession()
第一次调用是:创建 Session 会话之后调用都是:获取前面创建好的 Session 会话对象。
isNew(); 判断到底是不是刚创建出来的(新的)
true 表示刚创建false 表示获取之前创建
每个会话都有一个身份证号。也就是 ID 值。而且这个 ID 是唯一的。
getId() 得到 Session 的会话 id 值。
servlet程序:
protected void createOrGetSession(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//创建和获取Session会话对象
HttpSession session = req.getSession();
//判断当前对象是否是新创建出来的
boolean isNew = session.isNew();
//获取Session会话的唯一标识id
String id = session.getId();
resp.getWriter().write("得到Session,他的id是:"+id+"
");
resp.getWriter().write("这个Session是否是新创建的"+isNew+"
");
}
重新运行,需要清理之前浏览器上的cookie记录
第一次:
第N次
protected void setAttribute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.getSession().setAttribute("key1", "value1");
resp.getWriter().write("已经往Session中保存了数据");
}
protected void getAttribute(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
Object attribute = req.getSession().getAttribute("key1");
resp.getWriter().write("从Session中获取出key1的数据是" + attribute);
}
4、Session 生命周期控制
public void setMaxInactiveInterval(int interval) 设置 Session 的超时时间(以秒为单位),超过指定的时长,Session就会被销毁。
值为正数的时候,设定 Session 的超时时长。负数表示永不超时(极少使用)
public int getMaxInactiveInterval()获取 Session 的超时时间
public void invalidate() 让当前 Session 会话马上超时无效。
Session 默认的超时时长是多少!
Session 默认的超时时间长为 30 分钟。
因为在 Tomcat 服务器的配置文件 web.xml中默认有以下的配置,它就表示配置了当前 Tomcat 服务器下所有的 Session
超时配置默认时长为:30 分钟。
30
(1)如果说。你希望你的 web 工程,默认的 Session 的超时时长为其他时长。你可以在你自己的 web.xml 配置文件中做以上相同的配置。就可以修改你的 web 工程所有 Seession 的默认超时时长。
(2)如果你想只修改个别 Session 的超时时长。就可以使用上面的 API。setMaxInactiveInterval(int interval)来进行单独的设置。
session.setMaxInactiveInterval(int interval)单独设置超时时长。
protected void defaultLife(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获取了Session的默认超时时长
int maxInactiveInterval = req.getSession().getMaxInactiveInterval();
resp.getWriter().write("Session的默认超时时长为:" + maxInactiveInterval + " 秒 ");
}
protected void life3(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 先获取Session对象
HttpSession session = req.getSession();
// 设置当前Session3秒后超时
session.setMaxInactiveInterval(3);
resp.getWriter().write("当前Session已经设置为3秒后超时");
}
protected void deleteNow(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 先获取Session对象
HttpSession session = req.getSession();
// 让Session会话马上超时
session.invalidate();
resp.getWriter().write("Session已经设置为超时(无效)");
}
每次都要点击超时按钮再创建刷新
Session 超时的概念介绍:
Session 技术,底层其实是基于 cookie 技术来实现的。
其中cookie是客户端用于记录,session用于服务端记录
所以就需要cookie和session来存储这些信息【比如登陆的用户信息等】
- Filter 过滤器它是 JavaWeb 的三大组件之一。
三大组件分别是:Servlet 程序、Listener 监听器、Filter 过滤器Filter 过滤器它是 JavaEE 的规范。也就是接口Filter 过滤器它的作用是:拦截请求,过滤响应。
拦截请求常见的应用场景有:
5. 权限检查
6. 日记操作
7. 事务管理
……等等
要求:在你的 web 工程下,有一个 admin 目录。这个 admin 目录下的所有资源(html 页面、jpg 图片、jsp 文件、等等)都必须是用户登录之后才允许访问。
思考:根据之前我们学过内容。我们知道,用户登录之后都会把用户登录的信息保存到 Session 域中。所以要检查用户是否登录,可以判断 Session 中否包含有用户登录的信息即可!!!
工作流程
Filter 的代码:
public class AdminFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
HttpSession session = httpServletRequest.getSession();
Object user = session.getAttribute("user");
//如果等于null,说明还没有登录
if (user == null){
servletRequest.getRequestDispatcher("/login.jsp").forward(servletRequest,servletResponse);
return;
}else{
//让用户继续向下访问用户的目标资源
filterChain.doFilter(servletRequest,servletResponse);
}
}
}
web.xml 中的配置
AdminFilter
com.cen.Filter.AdminFilter
AdminFilter
public static void main(String[] args) {
//1.创建小窗口对象(被监听的对象)
Myframe myframe = new Myframe();
//设置窗口宽高
myframe.setBounds(0,0,200,200);
//2.设置窗口显示
myframe.setVisible(true);
//在事件源上绑定监听器
myframe.addWindowListener(new MyWindowListener());
}
}
class MyWindowListener implements WindowListener{
@Override
public void windowOpened(WindowEvent e) {
}
@Override
public void windowClosing(WindowEvent e) {
System.out.println("窗口关闭");
System.exit(0);
}
@Override
public void windowClosed(WindowEvent e) {
}
@Override
public void windowIconified(WindowEvent e) {
}
@Override
public void windowDeiconified(WindowEvent e) {
}
@Override
public void windowActivated(WindowEvent e) {
}
@Override
public void windowDeactivated(WindowEvent e) {
}
}
GUI窗口
控制台
在Servlet中定义了多种类型的监听器,他们用于监听事件源分别是ServletContext、HttpSession、ServletRequest这三个域对象
(2)监听器的分类- 监听三个域对象的创建和销毁的监听器(三个)监听三个域对象的属性变更(属性添加、移除、替换)的监听器(三个)监听HttpSession中JavaBean的状态改变(钝化、活化、绑定、解除绑定)的监听(两个)
用来监听ServletContext域对象的创建和销毁的监听器
②ServletContext创建和销毁创建:在服务器启动的时候,为每一个web应用创建单独的ServletContext对象
销毁:在服务器关闭的时候,或者项目从web服务器中关闭的时候
③ServletContext创建和销毁监听ServletContext对象的创建
监听ServletContext对象的销毁
java代码
public class MyServletContextListener implements ServletContextListener {
@Override
//监听ServletContext对象的创建
public void contextInitialized(ServletContextEvent servletContextEvent) {
System.out.println("ServletContext对象被创建");
}
@Override
//监听ServletContext对象的销毁
public void contextDestroyed(ServletContextEvent servletContextEvent) {
System.out.println("ServletContext对象被销毁");
}
}
配置
④ServletContextListener企业用途com.cen.webListener.MyServletContextListener
加载配置文件:Spring框架提供了一个核心监听器ContextListener
定时任务调度:在服务器启动的时候去计时,当到达某个时间点时可以去执行某个程序
用来监听HttpSession对象的创建和销毁
②HttpSessionListener创建和销毁创建:服务器端第一次调用getSession()方法的时候
访问Html是否创建Session:不会访问jsp是否创建Session:会访问Servlet是否创建Session:不会(默认没有调用getSession()方法)
销毁:
1、非正常关闭服务器(正常关闭服务器session会被序列化生成一个.ser文件)
2、Session过期(默认过期时间30min)
3、手动调用session.invalidate()方法
监听HttpSession的创建
监听HttpSession的销毁
public class MyHttpSessionListener implements HttpSessionListener {
@Override
//监听HttpSession的创建
public void sessionCreated(HttpSessionEvent httpSessionEvent) {
System.out.println("HttpSession对象被创建");
}
@Override
//监听HttpSession的销毁
public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
System.out.println("HttpSession对象被销毁");
}
}
配置
(3)ServletRequestListener监听器 ①ServletRequestListener监听器的作用com.cen.webListener.MyHttpSessionListener
用来监听ServletRequest对象的创建和销毁
②ServletRequestListener创建和销毁只要有请求且有响应,则创建和销毁都会同时发生
创建:从客户端向服务器发送一次请求,服务器就会创建Request对象。
访问Html是否创建Session:会访问jsp是否创建Session:会访问Servlet是否创建Session:不会
销毁:服务器对这次请求作出相应了之后,Request对象就销毁了
③编写监听器监听ServletRequest对象创建和销毁public class ServletRequestListener implements javax.servlet.ServletRequestListener {
@Override
//ServletRequest对象被销毁
public void requestDestroyed(ServletRequestEvent servletRequestEvent) {
System.out.println("ServletRequest对象被销毁");
}
@Override
//ServletRequest对象被创建
public void requestInitialized(ServletRequestEvent servletRequestEvent) {
System.out.println("ServletRequest对象被创建");
}
}
配置
(4)统计当前在线人数com.cen.webListener.ServletRequestListener
分析: 当用户登录时,实际上是由客户端向服务器发送一次请求,当服务器接收到请求后会为每一个用户建立一个session(会话),存储识别该用户的一些信息例如用户名等,这样当该用户浏览别的网页时不需要重新登录,因为session中已经有此用户信息,而且当该用户再次发送请求时,session会话仍是那一个,所以统计用户在线人数,可通过对session对象建立加以监听来实现,而用户数目需要存储在全局变量ServletContext对象中.
onlineCountServletContextListener类中
public class onlineCountServletContextListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
//在服务器启动的时候初始化一个值为0
//还要将这个值存到ServletContext中
servletContextEvent.getServletContext().setAttribute("count",0);
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
}
}
onlineCountHttpSessionListener类中
public class onlineCountHttpSessionListener implements HttpSessionListener {
@Override
public void sessionCreated(HttpSessionEvent httpSessionEvent) {
//上线了
HttpSession httpSession = httpSessionEvent.getSession();
System.out.println(httpSession.getId()+"上线了");
//获得ServletContext中的值
Integer count = (Integer) httpSession.getServletContext().getAttribute("count");
count++;
httpSession.getServletContext().setAttribute("count",count);
}
@Override
public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
//离线了
HttpSession httpSession = httpSessionEvent.getSession();
System.out.println(httpSession.getId()+"离线了");
//获得ServletContext中的值
Integer count = (Integer) httpSession.getServletContext().getAttribute("count");
count--;
httpSession.getServletContext().setAttribute("count",count);
}
}
配置
com.cen.onlineCount.onlineCountHttpSessionListener com.cen.onlineCount.onlineCountServletContextListener
显示界面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
在线人数:${count}
(二) 监听三个域对象的属性变更(属性添加、移除、替换)的监听器(三个)
(1)ServletContextAttributeListener
监听ServletContext对象中的属性变更(属性添加、移除、替换)的监听器
监听HttpSession对象中的属性变更(属性添加、移除、替换)的监听器
监听ServletRequest对象中的属性变更(属性添加、移除、替换)的监听器
public class MyHttpSessionAttributeListener implements HttpSessionAttributeListener {
@Override
public void attributeAdded(HttpSessionBindingEvent httpSessionBindingEvent) {
System.out.println("向Session中添加了属性");
}
@Override
public void attributeRemoved(HttpSessionBindingEvent httpSessionBindingEvent) {
System.out.println("向Session中移除了属性");
}
@Override
public void attributeReplaced(HttpSessionBindingEvent httpSessionBindingEvent) {
System.out.println("向Session中替换了属性");
}
}
配置
com.cen.webListener.MyHttpSessionAttributeListener
jsp文件
概述
保存在Session域中的Java类可以有多种状态:绑定到Session中;从Session中解除绑定;随Session对象持久化到一个存储设备中(钝化);随Session对象从一个存储设备中恢复(活化)。
Servlet对象中定义了两个特殊的监听的接口来帮助Java类了解自己在Session域中的状态:
HttpSessionBindingListener接口HttpSessionActivationListener接口
实现这两个接口的类不需要在web.xml中配置
(1)HttpSessionBindingListener监听器
监听Java类在HttpSession中的绑定和解除绑定的状态的监听器
测试代码
public class Bean1 implements HttpSessionBindingListener {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public void valueBound(HttpSessionBindingEvent httpSessionBindingEvent) {
System.out.println("Bean1与session绑定了");
}
@Override
public void valueUnbound(HttpSessionBindingEvent httpSessionBindingEvent) {
System.out.println("Bean1与session解除绑定了");
}
}
jsp文件中
监听HttpSession中Java类的钝化和活化监听器
public class Bean2 implements HttpSessionActivationListener, Serializable {
private String NAME;
public String getNAME() {
return NAME;
}
public void setNAME(String NAME) {
this.NAME = NAME;
}
@Override
//钝化:相当于被序列化,被写入硬盘了
public void sessionWillPassivate(HttpSessionEvent httpSessionEvent) {
System.out.println("Bean2被Session钝化了");
}
@Override
public void sessionDidActivate(HttpSessionEvent httpSessionEvent) {
System.out.println("Bean2被Session活化了");
}
}
序列化过程
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
<%
Bean2 bean2 = new Bean2();
bean2.setNAME("bean2");
session.setAttribute("bean2",bean2);
%>
页面显示活化
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
${bean2.NAME}
操作过程:
启动服务器
访问bean2.jsp执行序列化过程
访问bean22.jsp查看页面
正常关闭服务器------》执行打印被钝化
启动服务器
直接访问bean22.jsp仍有信息-------》被活化
配置完成session序列化和反序列化(自动序列化和反序列化,模拟用户登录记录)
Context标签可以配置在:
tomcat/conf/context.xml : 所有tomcat下虚拟主机和虚拟目录下的工程都会序列化sessiontomcat/conf/Catalina/localhost/context.xml : localhost虚拟主机下的所有项目会序列化session工程/meta-INF/context.xml : 当前工程目录下才会序列化session
注意:!!!!!
①在tomocat中要√上该选项,否则无法活化
(意思是在重新启动和重新部署保存会话)
否则每次重启idea都会把原先会话删去
②Eclipse会自动创建meta-INF目录,但IDEA不会自动创建,需要我们需要手动在web下创建meta-INF,meta-INF和WEB-INF同一级别



