第一章-会话的概念
1.1 会话的概念
会话的概念
会话:浏览器和服务器之间的多次请求和响应。
用户打开浏览器,连接服务器,浏览不同的网页(资源),发出多个请求,直到关闭浏览器的过程,称为一次会话(多次请求).
我们在会话的过程(多次请求)之中,用户可能会产生一些数据,这些数据话有的需要保存起来的,我们就可以通过会话技术来保存用户各自的数据
为什么要使用会话技术
保存一次通话过程中的数据!
会话技术中数据的保存方式:
浏览器存储会话数据:cookie
服务器存储会话数据:session
常用的会话技术
cookie
记住用户名密码
cookie是客户端(浏览器)的技术,用户浏览的信息以键值对(key=value)的形式保存在浏览器上。如果没有关闭浏览器,再次访问服务器,会把cookie带到服务端,服务端就可以做相应的处理。
session
记住登录用户名,购物车
==session是服务器端的技术。==服务器为每一个浏览器开辟一块内存空间,即session。由于内存空间是每一个浏览器独享的,所有用户在访问的时候,可以把信息保存在session对象中。同时,每一个session对象都对应一个sessionId,服务器把sessionId写到cookie中,再次访问的时候,浏览器会把cookie(sessionId)带过来,找到对应的session对象。
第二章-Cookie
2.1 Cookie的概念和作用
Cookie的概念
Cookie是一种客户端的会话技术,它是服务器存放在浏览器的一小份数据,浏览器以后每次访问该服务器的时候都会将这小份数据携带到服务器去。
Cookie的作用
- 在浏览器中存放数据
- 将浏览器中存放的数据携带到服务器
Cookie的应用场景
1.记住用户名
当我们在用户名的输入框中输入完用户名后,浏览器记录用户名,下一次再访问登录页面时,用户名自动填充到用户名的输入框.
2.自动登录(记住用户名和密码)
当用户在淘宝网站登录成功后,浏览器会记录登录成功的用户名和密码,下次再访问该网站时,自动完成登录功能.
以上这些场景都是使用会话cookie实现的,将上次的信息保存到了cookie中,下次直接从cookie中获取数据信息
3.保存网站的上次访问时间
我们访问网站的时候,通常会看到网站上显示上次访问时间,这些信息就是在用户访问网站的时候保存在cookie中的
4.保存电影的播放进度
在网页上播放电影的时候,如果中途退出浏览器了,下次再打开浏览器播放同一部电影的时候,会自动跳转到上次退出时候的进度,因为在播放的时候会将播放进度保存到cookie中
2.2 Cookie的快速入门
相关的API
new Cookie(String name,String value); 创建一个Cookie对象(cookie只能保存字符串数据。可以保存中文 但是要求tomcat版本8.5+)
response.addCookie(cookie对象); 把cookie写入浏览器
Cookie[] request.getCookies() ;获得浏览器带过来的所有Cookie:得到所有的cookie对象。是一个数组,开发中根据key得到目标cookie中保存的值
cookie.getName() ; 获得cookie中设置的key
cookie.getValue(); 获得cookie中设置的value
入门代码
@WebServlet("/ServletDemo1")
public class ServletDemo1 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取浏览器携带的所有cookie
Cookie[] cookies = request.getCookies();
// 循环遍历所有的cookie
if (cookies != null) {
for (Cookie cookie : cookies) {
String name = cookie.getName();
String value = cookie.getValue();
System.out.println("cookie的键:" + name + ",cookie的值" + value);
}
}
// 1.创建Cookie对象,并存储数据
Cookie aCookie = new Cookie("akey", "aaa");
Cookie bCookie = new Cookie("bkey", "bbb");
// 2.把Cookie对象响应给浏览器
response.addCookie(aCookie);
response.addCookie(bCookie);
// 3.响应数据到页面
response.getWriter().print("ServletDemo1...");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
cookie的特点
- cookie是一门客户端会话技术,保存数据在客户端浏览器
- cookie在第一次请求时没有,通过服务器以响应头的方式写给浏览器,在浏览器第二次发送 请求时,就会携带cookie到服务器
- cookie存储的字符串数据
- cookie存储的大小以及个数有限
- cookie存储数据不安全 因此cookie会存储一些私人数据或不重要的数据
中文乱码问题
URLEncode.encode(value,"utf-8");//存入的时候(先通过utf-8编码)
URLDecode.decode(value,"utf-8");//取出 (通过utf-8解码)
@WebServlet("/ServletDemo2")
public class ServletDemo2 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取浏览器携带的所有cookie
Cookie[] cookies = request.getCookies();
// 循环遍历所有的cookie
if (cookies != null) {
for (Cookie cookie : cookies) {
String name = cookie.getName();
String value = cookie.getValue();
System.out.println("cookie的键:" + URLDecoder.decode(name,"utf-8") + ",cookie的值:" + value);
}
}
// 1.创建Cookie对象,并存储数据
Cookie aCookie = new Cookie(URLEncoder.encode("中国","utf-8"), "hello");
// 2.把Cookie对象响应给浏览器
response.addCookie(aCookie);
// 3.响应点数据到页面
response.getWriter().print("ServletDemo2...");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
封装工具类(获得指定的cookie)
我们一般根据cookie的name获得目标cookie对象. 我们可以把这块逻辑封装到工具类里面
-
工具类
public class CookieUtils { /** * 查找指定的cookie * @param cookieName * @param cookies * @return cookie对象 */ public static Cookie getCookie(String cookieName,Cookie[] cookies){ // 如果没有携带cookie if (cookies == null){ return null; } // 如果有携带cookie,找到了 for (Cookie cookie : cookies) { String name = cookie.getName(); if (cookieName.equals(name)){ // 找到了 return cookie; } } // 如果有携带cookie,没找到 return null; } }
-
测试工具类
@WebServlet("/ServletDemo3") public class ServletDemo3 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 需求: 查找名字为akey的cookie----->先访问ServletDemo1(写aCookie,bCookie到浏览器),然后访问ServletDemo3 // 1.获得浏览器携带的所有Cookie Cookie[] cookies = request.getCookies(); // 2.查找名字为akey的cookie Cookie aCookie = CookieUtils.getCookie("akey", cookies); // 3.打印cookie的名字和值 System.out.println(aCookie.getName() + " = " + aCookie.getValue()); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } }
2.3 Cookie进阶
cookie的分类(有效时长)
- 会话级别cookie
在默认的情况下,当浏览器进程结束(浏览器关闭,会话结束)的时候,cookie就会消失。
-
持久性cookie
给cookie设置有效期.
cookie.setMaxAge(int expiry)
:参数是时间,单位是秒 -1:默认。代表Cookie数据存到浏览器关闭(保存在浏览器文件中)。
正整数:以秒为单位保存数据有有效时间(把缓存数据保存到磁盘中)
0:代表删除Cookie.如果要删除Cookie要确保路径一致。
@WebServlet("/ServletDemo4")
public class ServletDemo4 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取所有的cookie对象,打印数据
Cookie[] cookies = request.getCookies();
if (cookies != null){
for (Cookie cookie : cookies) {
System.out.println(cookie.getName()+"="+cookie.getValue());
}
}
// 创建Cookie对象
Cookie aCookie = new Cookie("akey", "aaa");
Cookie bCookie = new Cookie("bkey", "bbb");
// 设置aCookie的有效时长,bCookie的有效时长为默认
aCookie.setMaxAge(60*60*24*7);
// 响应Cookie对象
response.addCookie(aCookie);
response.addCookie(bCookie);
// 响应数据到页面,证明当前Servlet执行了
response.getWriter().print("ServletDemo4....");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
cookie设置有效路径
setPath(String url) ;设置路径,如果不设置路径,就是默认路径
有效路径作用 :
- 保证不会携带别的网站/项目里面的cookie到我们自己的项目(一个是域名ip地址隔离,一个路径隔离)
- 如果路径不一样, cookie的name可以相同
- 保证自己的项目可以合理的利用自己项目的cookie(默认路径: /项目部署路径 eg:/day24)
-
默认路径,例如:
- 访问http://localhost:8080/day25/ServletDemo5; cookie默认路径 /day25
-
访问http://localhost:8080/day25/aa/ServletDemo6; cookie默认路径 /day25/aa
-
访问http://localhost:8080/day25/aa/bb/ServletDemo7; cookie默认路径 /day25/aa/bb
-
…
-
携带Cookie需要的条件(设置有效路径): 只有当访问资源的url上一级目录与此cookie的有效path完全匹配的时候,才会携带这个cookie; url上一级目录如果是cookie有效路径的子目录也可以获取到反之不会.
- eg: 假设Cookie路径是: /day25
- 访问http://localhost:8080/day25/ServletDemo8; 会携带该cookie过来
-
访问http://localhost:8080/day25/aa/ServletDemo9; 会携带该cookie过来
-
访问http://localhost:8080/day25/aa/bb/ServletDemo10; 会携带该cookie过来
-
…
-
eg: 假设Cookie路径是: /day25/aa
-
访问http://localhost:8080/day25/ServletDemo11; 不会携带该cookie过来
-
访问http://localhost:8080/day25/aa/ServletDemo12; 会携带该cookie过来
-
访问http://localhost:8080/day25/aa/bb/ServletDemo13; 会携带该cookie过来
-
…
-
Cookie小结
-
有效路径作用 :
- 保证不会携带别的网站/项目里面的cookie到我们自己的项目
- 保证自己的项目可以合理的利用自己项目的cookie
- 如果路径不一样, cookie的key可以相同
- 如果路径一样, cookie的key相同,那么后面的Cookie会覆盖前面的Cookie
-
cookie的路径通常设置为项目的部署路径. 当前项目下的Servlet都可以使用该cookie. 一般这么设置: cookie.setPath(request.getContextPath());
只要是当前项目里面的资源 路径必须包含项目名路径.
-
案例-记录用户各自的上次访问时间
1.需求
在访问一个资源的时候,展示上次访问的时间
若是第一次访问则展示:你是第一次访问,若不是第一次则展示:你上次访问的时间是:xxxx
2.分析
1.获取浏览器携带的所有Cookie---->数组cookies
2.判断是否是第一次访问(其实就是判断携带的Cookie里面有没有目标Cookie)
3.如果是第一次访问:
3.1 创建当前日期对象
3.2 把日期对象转换为对应的毫秒值( long getTime())
3.3 创建Cookie对象,存储当前访问的系统时间
3.4 设置有效时长
3.5 设置有效路径(项目部署路径)
3.6 响应给浏览器
3.7 响应欢迎数据到页面(欢迎第一次访问本网站)
4.如果不是第一次访问:
4.1 获取存储上一次访问时间的目标Cookie对象
4.2 获取目标Cookie对象中的时间
4.3 把毫秒值时间转换为Date日期
4.4 把当前系统时间存储到目标Cookie中(更新时间)
4.5 把时间响应到页面上(您上一次访问的时间是: xxx...)
3.代码实现
@WebServlet("/ServletDemo16")
public class ServletDemo16 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//请求响应中文乱码处理
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
Cookie[] cookies = request.getCookies();
//1.根据cookie的name【lastTime】获取上次访问时间
Cookie lastTime = CookieUtils.getCookieValue(cookies, "lastTime");
//2.判断value
if(lastTime==null){
//3.1:value==null:第一次访问
//获取当前时间,记录在cookie中,写入到浏览器
String dateStr = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date());
Cookie cookie = new Cookie("lastTime", URLEncoder.encode(dateStr,"UTF-8"));
response.addCookie(cookie);
//响应浏览器 用户 你是第一次访问
response.getWriter().print("你是第一次访问!");
}else{
//3.2:value!=null:显示上次访问时间
//获取当前时间,记录在cookie中,写入到浏览器
String dateStr = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date());
Cookie cookie = new Cookie("lastTime", URLEncoder.encode(dateStr,"UTF-8"));
response.addCookie(cookie);
//响应浏览器 用户 你上次访问的时间
response.getWriter().print("你上次访问时间是:"+ URLDecoder.decode(lastTime,"UTF-8"));
}
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
4.小结
-
案例关键: 判断是否第一次访问【说白了就是判断目标Cookie是否为null】
-
取的时候 和存时候的key要一致
-
不管是哪一次访问, 都需要记录当前的时间到Cookie
第三章-Session
3.1 session概述
session概述
session是服务器端的技术。服务器为每一个浏览器开辟一块内存空间,即session对象。由于session对象是每一个浏览器特有的,所以用户的记录可以存放在session对象中。同时**,每一个session对象都对应一个sessionId,服务器把sessionId写到cookie中,再次访问的时候,浏览器把sessionId带过来,找到对应的session对象**(session是基于cookie的)
cookie和Session的不同
- cookie是保存在浏览器端的,大小和个数都有限制。session是保存在服务器端的, 原则上大小是没有限制(实际开发里面也不会存很大的数据), 安全一些。
- cookie不支持中文,并且只能存储字符串;session可以存储基本数据类型,集合,对象等
- session是基于cookie实现的
Session的执行原理
1. 第一次请求,通过request.getSession()创建session对象
2. 创建完session对象时,会生成sessionId,保存到cookie中响应给客户端浏览器
3. 第二次请求时,客户端浏览器会携带sessionId到服务器,服务器就根据sessionId找到对应的session对象使用
1.浏览器请求服务器,会携带cookie
2.服务器就会从cookie中获取sessionId
3.判断是否有sessionId,
4.如果没有,就直接为该浏览器创建新的Session对象(request.getSession() —>没有就创建,有就获取)
5.如果有sessionId,就根据sessionId查找对应的session对象
6.如果能找到session对象,那就直接使用
7.如果找不到session对象(session对象销毁),就直接创建新的session对象
注意:
- 第一次调用request.getSession()才会创建session对象,不调用则session对象不存在
- session对象的生命周期范围就是从第一次调用request.getSession(),session对象被创建,当关闭浏览器,session对象被持久化到硬盘
- session对象在创建时,会自动产生sessionId,由服务器自动通过cookie方式写入到浏览器缓存中,该cookie的name是JSESSIONID
@WebServlet("/ServletDemo17")
public class ServletDemo17 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.获得session对象
HttpSession session = request.getSession();
// 2.获得sessionId
String id = session.getId();
System.out.println("sessionId:" + id);
// 3.响应数据到页面,证明该Servlet执行了
response.getWriter().print("ServletDemo17...");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
3.2 Session作为域对象存取值
Session作为域对象存取值
范围: 一次会话(多次请求) 保存用户各自的数据(以浏览器为单位)
- request.getSession(); 获得session(如果第一次调用的时候其实是创建session,第一次之后通过sessionId找到session进行使用)
- Object getAttribute(String name) ;获取值
- void setAttribute(String name, Object value) ;存储值
- void removeAttribute(String name) ;移除值,只是移除键值对,并不会删除session对象
@WebServlet("/ServletDemo18")
public class ServletDemo18 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 存值
request.getSession().setAttribute("akey","aaa");
response.getWriter().print("ServletDemo18...set...");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
@WebServlet("/ServletDemo19")
public class ServletDemo19 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.处理乱码
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
// 取值
String value = (String) request.getSession().getAttribute("akey");
response.getWriter().print("取出的session值:"+value);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
@WebServlet("/ServletDemo20")
public class ServletDemo20 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 删除值----->并不会删除session对象,至少把值给干掉了
request.getSession().removeAttribute("akey");
response.getWriter().print("ServletDemo20...remove...");
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
session有效期
方式一:tomcat设置session失效时间,默认30min
conf---->web.xml
<session-config>
<session-timeout>30</session-timeout>
</session-config>
方式二:设置失效时间 setMaxInactiveInterval(); 单位是秒
方式三:手动设置失效 invalidate();
@WebServlet("/session03")
public class SessionServlet03 extends HttpServlet {
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("SessionServlet03...");
//获取session中的共享数据
HttpSession session = request.getSession();
//设置session失效
//方式二:设置失效时间 单位是秒 正数表示有效时间 负数表示永远不会过时
//session.setMaxInactiveInterval(20);
//方式三:手动设置失效 立即失效
session.invalidate();
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
session持久化
-
浏览器关闭,会话结束,此时session对象有没有被销毁?没有
session对象虽然还存在,但是找不到了!因为新打开浏览器发起请求,产生了一个新的sessionId,找不到 原来的session对象了。 -
如果服务器正常关闭,浏览器没有关闭,session对象有没有被销毁?还能不能获取到session中存储的数据?
session钝化:指的是服务器正常关闭,会自动将session数据持久化保存到服务器硬盘的文件上。
session活化:指的是服务器正常启动,会将持久化的session数据加载到内存中,重新创建一个
session对象,存储活化后的数据,但是sessionId没有改变。
tomcat\Tomcat_8_5_27\work\Catalina\localhost
目录查看注意:session的钝化和活化是服务器自动完成的,我们了解服务器的这种机制即可。
关于Session对象的销毁
-
浏览器关闭了, session使用不了, 是session销毁了吗?
- session没有销毁,只是浏览器端的含有sessionId的cookie没有了.
- session基于cookie, sessionId保存到cookie里面的, 默认情况下cookie是会话级别,浏览器关闭了cookie就是消失了,也就是说sessionId消失了, 从而找不到对应的session对象了, 就不能使用了.
-
结论:
- 1.服务器响应的sessionId,是使用的默认基本的Cookie,关闭浏览器,cookie销毁了,也就是sessionId没有了,但session对象还在服务器中
- 2.服务器响应的sessionId是持久级别的Cookie,关闭浏览器,cookie还在,也就是sessionId还在,并且session对象也还在服务器中, 可以获取上一次会话保存在session中的数据
-
**解决: 自己获得sessionId, 自己写给浏览器 设置Cookie的有效时长, 这个Cookie的key必须:
JSESSIONID**
@WebServlet("/ServletDemo21") public class ServletDemo21 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 创建session对象存值 HttpSession session = request.getSession(); // 获取sessionId String id = session.getId(); System.out.println("ServletDemo21...id:"+id); // 存值 session.setAttribute("akey","aaa"); // 手动响应sessionId Cookie cookie = new Cookie("JSESSIONID",id); // 设置有效时长 cookie.setMaxAge(60*60); // 设置有效路径 cookie.setPath(request.getContextPath()); // 响应cookie response.addCookie(cookie); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } }
-
真正的销毁session对象,得使用session对象的invalidate()方法
@WebServlet("/ServletDemo23") public class ServletDemo23 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // 销毁session对象 request.getSession().invalidate(); response.getWriter().println("ServletDemo23...销毁了session对象..."); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doPost(request, response); } }
3.3 三个域对象比较
三个域对象比较
域对象 | 创建 | 销毁 | 作用范围 | 应用场景 |
---|---|---|---|---|
ServletContext | 服务器启动 | 服务器正常关闭/项目从服务器移除 | 整个项目 | 记录访问次数,聊天室 |
HttpSession | 第一次调用request.getSession()方法 | session过期(默认30分钟)/调用invalidate()方法/服务器异常关闭 | 会话(多次请求) | 验证码校验, 保存用户登录状态等 |
HttpServletRequest | 来了请求 | 响应这个请求(或者请求已经接收了) | 一次请求 | servletA和jsp(servletB)之间数据传递(转发的时候存数据) |
三个域对象怎么选择?
三个域对象怎么选择?
- 常规选择:一般使用范围最小的能完成就选择最小的
- 具体请求选择:
- 转发:使用request作用域 重定向:使用session
- 查询:使用request作用域
- 根据范围:一次请求:【request】 一次会话:【session】 整个应用:【ServletContext】
案例-一次性验证码校验
1.需求
在网站登录的时候,生成一个验证码.登录的时候对验证码进行校验.
2.实现
2.1生成验证码
- 拷贝验证码的jar包
- 创建CodeServlet
//1.生成验证码
//2.响应给客户端(浏览器)
@WebServlet("/code")
public class CodeServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1.生成验证码
ValidateCode validateCode = new ValidateCode(200, 50, 4, 20);
//2.将验证码存入到session中
String code = validateCode.getCode();
request.getSession().setAttribute("validateCode",code);
//3.将验证码响应到客户端浏览器显示
validateCode.write(response.getOutputStream());
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
2.2校验验证码
@WebServlet("/login")
public class LoginServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
try {
// 1.请求响应中文乱码处理
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
// 2.获取请求参数【用户名、密码、验证码】
String username = request.getParameter("username");
String password = request.getParameter("password");
String code = request.getParameter("code");
// 3.判断用户输入的验证码和后台生成的验证码是否一致
//获取后台生成的验证码
String validateCode = request.getSession().getAttribute("code").toString();
if(code!=null && !code.equals("") && code.equalsIgnoreCase(validateCode)){
// 3.1:一致 判断根据用户名和密码是否可以查询到指定的user
QueryRunner queryRunner = new QueryRunner(C3P0Utils.getDataSource());
String sql ="select * from user where username=? and password=?";
User user = queryRunner.query(sql, new BeanHandler<>(User.class), username, password);
if(user!=null){
//user!=null:登录成功
response.getWriter().print("登录成功!");
}else{
//user==null:登录失败
response.getWriter().print("登录失败!");
}
}else{
// 3.2:不一致:验证码输入有误
response.getWriter().print("验证码输入有误!");
}
} catch (Exception e) {
e.printStackTrace();
response.getWriter().print("服务器异常!");
}
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<center>
<h1>用户登录</h1>
<form action="login" method="post">
用户名:<input type="text" name="username"/><br/>
密码:<input type="password" name="password"/><br/>
验证码:<input type="text" name="code"><br>
<!-- code 等价于 http://localhost:8080/虚拟路径/code this表示的就是当前img标签对象-->
<img src="code" alt="" onclick="changeImg(this)"><br>
<input type="submit" value="登录"/>
</form>
</center>
<script>
function changeImg(obj) {
console.log("重新加载验证码图片");
//由于请求地址一样时,不再发送新的请求 所以在这里,需要通过参数的改变,去忽悠浏览器,有一个新的请求到来,重新生成新的验证码响应
//需要 每点击一次 请求地址参数就刷新一次 才会向服务器发送新的请求
obj.src="code?timestr="+new Date().getTime();
}
</script>
</body>
</html>
3.小结
- 需要在CodeServlet 把生成的验证码存到session里面
- 不能存到ServletContext, 就共享验证码了
- 不能存到request, 根本不能用
-
思路
- CodeServlet: 生成验证码存到Session
- LoginServlet:
- 获得用户输入的验证码
- 获得session里面存的验证码
- 比较是否一致
第四章_JSP入门
4.1 JSP概述
什么是JSP
Java server page(java服务器页面). JSP本质就是Servlet
它和servlet技术一样,都是SUN公司定义的一种用于开发动态web资源的技术。
JSP=html(js,css)+java+jsp特有的内容
JSP产生的原因
需求: 我们要向页面动态输出一个表格. 发现特别的繁琐
servlet在展示页面的时候,相当的繁琐。sun公司为了解决这个问题,参照asp开发了一套动态网页技术jsp。
JSP执行原理
JSP会翻译(通过默认的JspServlet,JSP引擎)成Servlet(.java),Servlet编译成class文件
JSP执行流程
第一次访问的xxx.jsp时候,服务器收到请求,JspServlet会去查找对应的jsp文件
找到之后,服务器会将这个jsp文件转换成java文件(Servlet)
服务器编译java文件,生成class文件
服务器运行class文件,生成动态的内容
服务器收到内容之后,返回给浏览器
3.小结
- JSP:是一个动态网页,在jsp中既可以写java代码,又可以写html代码,作用:简化开发
- JSP本质就是一个Servlet
- JSP执行流程:请求–>index.jsp-翻译->index_jsp.java-编译->index_jsp.class–>执行–>响应给浏览器
- 在目前实际开发中,jsp基本不再使用了,1.执行效率低下 2.不利于分工协作
4.2 JSP基本语法
JSP脚本
我们可以通过JSP脚本在JSP页面上编写Java代码. 一共有三种方式:
类型 | 翻译成Servlet对应的部分 | 注意 |
---|---|---|
<%…%>:Java程序片段 | 翻译成Service()方法里面的内容, 局部的 | |
<%=…%>:输出表达式 | 翻译成Service()方法里面的内容,相当于调用out.print() | 输出表达式不能以;结尾 |
<%!..%>:声明成员变量 | 翻译成Servlet类里面的内容 |
- eg
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>$Title$</title>
</head>
<body>
<%!
//声明成员变量或方法 会被翻译到Servlet类中
String hehe = "hello";
%>
<%
//编写java代码
System.out.println("Hello world!");
//声明一个变量 局部变量 会被翻译到service方法中
String msg = "Hello";
%>
<h1>Hello World</h1>
<%--输出表达式:用于输出数据到页面--%>
<h2><%=msg %></h2>
<h3><%=hehe %></h3>
</body>
</html>
JSP注释
注释类型 |
---|
HTML注释 |
JAVA注释 //; /* */ |
JSP注释; <%–注释内容–%> |
注释快捷键:Ctrl+Shift+/ctrl+=
3.小结
- 脚本
- <% java代码 %>:java小脚本 会被翻译到service方法中
- <%=变量 %>:输出表达式 用于输出内容到页面 会被翻译到service方法中
- <%! java代码 %>:声明 用于声明全局变量或方法 会被翻译到servlet类中
- 注释:
- jsp独有的注释:<%-- 注释内容 --%>
- 注意:JSP注释内容不会输出到页面,也不会显示在html的网页源代码中
案例-记住用户名案例
1.分析
-
准备一个登录页面
-
编写登录Servlet
-
登录成功的时候 判断记住用户名复选框有没有打钩
- 打钩:记住用户名 使用cookie保存在客户端 设置一个有效时间 7天
- 不打钩:不记住用户名 删除cookie
-
登录页面使用jsp,记住用户名之后,在jsp页面上使用java代码获取用户名显示
2.实现
- LoginServlet
package com.geekly.web;
import com.geekly.bean.User;
import com.geekly.utils.C3P0Utils;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.sql.SQLException;
@WebServlet("/login")
public class LoginServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
try {
//请求响应中文乱码处理
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
// 2.获取请求参数【用户名和密码】
String username = request.getParameter("username");
String password = request.getParameter("password");
// 3.使用DBUtils操作数据库 根据用户名和密码查询数据库,返回user对象
QueryRunner queryRunner = new QueryRunner(C3P0Utils.getDataSource());
String sql = "select * from user where username=? and password=?";
User user = queryRunner.query(sql,new BeanHandler<>(User.class),username,password);
// 4.根据返回处理结果进行响应 user==null:登录失败 user!=null:登录成功
if(user!=null){
String remember = request.getParameter("remember");
//登录成功
Cookie cookie = new Cookie("username", username);
if(remember!=null && remember.equals("on")){
//如果勾上记住用户名 将用户名保存在cookie中 并设置有效时间7天
cookie.setMaxAge(60*60*24*7);
}else{
//如果没有勾上记住用户名 就删除这个cookie
cookie.setMaxAge(0);
}
response.addCookie(cookie);
response.getWriter().print("登录成功!");
}else{
response.getWriter().print("登录失败!");
}
} catch (Exception e) {
e.printStackTrace();
response.getWriter().print("服务器异常!");
}
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
- login.jsp
<%@ page import="com.geekly.utils.CookieUtils" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>$Title$</title>
</head>
<body>
<center>
<h1>用户登录</h1>
<form action="login" method="post">
<%
String username = CookieUtils.getCookieValue(request.getCookies(),"username");
username = username==null?"":username;
%>
姓名:<input type="text" name="username" value="<%=username %>"/><br/>
密码:<input type="password" name="password"/><br/>
<input type="checkbox" name="remeber">记住用户名<br>
<input type="submit" value="登录"/>
</form>
</center>
</body>
</html>
3.小结
- 用户勾选了记住用户名,我们把用户名存到Cookie里面
- 在login.jsp里面 从cookie取出用户名展示
总结
1.记录用户上次访问的时间------>Cookie
2.记录用户名---->Cookie & jsp
3.一次性验证码校验---->Session
cookie的api和执行原理
session的api和执行原理
- 能够说出会话的概念
浏览器开启(发送多次请求)--->浏览器关闭
会话技术: cookie,session
- 能够创建、发送、接收、删除cookie
new Cookie(String name,String value);
response.addCookie(cookie);
request.getCookies() ;
cookie.getName() ; //返回cookie中设置的key
cookie.getValue(); //返回cookie中设置的value
- 能够完成记录用户各自的上次访问时间案例
1.第一次访问的时候,记录当前时间,保存到cookie中
2.第二次访问的时候,先获取之前创建的cookie,获取里面的时间,重新保存当前时间到cookie中,然后把cookie响应会去,再把获取的时间,显示到页面
....
- 能够获取session对象、添加、删除、获取session中的数据
request.getSession()
- Object getAttribute(String name) ;获取值
- void setAttribute(String name, Object value) ;存储值
- void removeAttribute(String name) ;移除
获取sessionId: session对象.getId();
- 能够完成登录验证码案例
1.在生成验证码的Servlet中生成验证码,并保存到session中
2.在登录的Servlet中获取用户输入的验证码和session保存的验证码进行对比
- 了解JSP的脚本
<% java代码:Servlet的service方法中 %>
<%= java代码:Servlet的service方法中的out.write()方法的小括号中 %>
<%! java代码:Servlet的类中 %>
注释: html,jsp,java三类注释
- 能够完成记住用户名案例
1.获取记住用户名复选框的值
2.判断获取的复选框的值
3.如果不为null,并且为指定的值(ok),那么就创建cookie对象,保存用户名,响应到浏览器
4.否则,就删除之前的cookie(设置之前cookie的有效时长为0---注意路径一定要一致)