您的位置:首页 > 编程语言 > Java开发

方立勋_30天掌握JavaWeb_Cookie

2017-03-30 21:29 399 查看

会话技术介绍

什么是会话?

会话可简单理解为:用户开一个浏览器,点击多个超链接,访问服务器多个web资源,然后关闭浏览器,整个过程称之为一个会话。

会话过程中要解决的一些问题?

每个用户在使用浏览器与服务器进行会话的过程中,不可避免各自会产生一些数据,程序要想办法为每个用户保存这些数据。

例如:用户点击超链接通过一个servlet购买了一个商品,程序应该想办法保存用户购买的商品,以便于用户点结帐servlet时,结帐servlet可以得到用户购买的商品为用户结帐。

思考:用户购买的商品保存在request或servletContext中行不行?

1. Cookie

Cookie是客户端技术,程序把每个用户的数据以cookie的形式写给用户各自的浏览器。当用户使用浏览器再去访问服务器中的web资源时,就会带着各自的数据去。这样,web资源处理的就是用户各自的数据了。



2. Session

Session是服务器端技术,利用这个技术,服务器在运行时可以为每一个用户的浏览器创建一个其独享的session对象,由于session为用户浏览器独享,所以用户在访问服务器的web资源时,可以把各自的数据放在各自的session中,当用户再去访问服务器中的其它web资源时,其它web资源再从用户各自的session中取出数据为用户服务。



Cookie

Cookie API:

javax.servlet.http.Cookie类用于创建一个Cookie,response接口也中定义了一个addCookie方法,它用于在其响应头中增加一个相应的Set-Cookie头字段。 同样,request接口中也定义了一个getCookies方法,它用于获取客户端提交的Cookie。Cookie类的方法:

public Cookie(String name,String value)

setValue与getValue方法

setMaxAge与getMaxAge方法

setPath与getPath方法 /day06

setDomain与getDomain方法

getName方法

Cookie应用

显示用户上次访问网站的时间

import java.io.IOException;
import java.io.PrintWriter;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

//代表网站首页
//Cookie技术显示用户上次访问网站的时间
public class CookieDemo1 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setCharacterEncoding("UTF-8");
response.setHeader("Content-type", "text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
out.print("<a href='/day07/servlet/CookieDemo2'>清除上次访问时间</a><br>");
out.print("您上次访问的时间是:");
// 拿到请求头中所有的cookie
Cookie cookies[] = request.getCookies();
// 遍历cookies找到lastAccessTime这个cookie
for (int i = 0; cookies != null && i < cookies.length; i++) {
if (cookies[i].getName().equals("lastAccessTime")) {
// 得到最后访问时间的毫秒数
long cookieValue = Long.parseLong(cookies[i].getValue());
// 将最后访问时间变成一个日期类型
Date date = new Date(cookieValue);
// 格式化这个日期
DateFormat df = new SimpleDateFormat("yyyy-MM-WW kk:mm:ss");
String lastAccessTime = df.format(date);

out.print(lastAccessTime);
}
}
// 将最新的访问时间设置回去
Cookie cookie = new Cookie("lastAccessTime", System.currentTimeMillis()
+ "");
// 设置有效期为一个月
cookie.setMaxAge(1 * 3600 * 24 * 30);
// 设置有效访问路径为/day07这个项目下的所有资源
cookie.setPath("/day07");
// 在response中增加一个cookie,服务器会自动增加Set-Cookie头字段
response.addCookie(cookie);
}

public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}

}


import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
//删除浏览器中的cookie
public class CookieDemo2 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
test1(response);

test2(request, response);
}

// 自己实验的方法,拿到了原有的cookie想要删除还是要先设置下Path,否则Path还是会在这个servlet的目录下
// 拿到了原有的cookie,如果只修改setMaxAge为0还是不行的!
private void test2(HttpServletRequest request, HttpServletResponse response) {
// 拿到请求头中所有的cookie
Cookie cookies[] = request.getCookies();
// 遍历cookies找到lastAccessTime这个cookie
for (int i = 0; cookies != null && i < cookies.length; i++) {
if (cookies[i].getName().equals("lastAccessTime")) {
Cookie cookie = cookies[i];
cookie.setMaxAge(0);
cookie.setPath("/day07");
response.addCookie(cookie);
}
}
}

// 开发中的方法
private void test1(HttpServletResponse response) {
Cookie cookie = new Cookie("lastAccessTime", "");
// 设置为0表示让浏览器删除这个cookie
cookie.setMaxAge(0);
// 删除cookie的时候Path必须一致,因为同一个cookie,Path肯定一致
cookie.setPath("/day07");
response.addCookie(cookie);
}

public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}

}


Cookie细节

一个Cookie只能标识一种信息,它至少含有一个标识该信息的名称(NAME)和设置值(VALUE)。

一个WEB站点可以给一个WEB浏览器发送多个Cookie,一个WEB浏览器也可以存储多个WEB站点提供的Cookie。

浏览器一般只允许存放300个Cookie,每个站点最多存放20个Cookie,每个Cookie的大小限制为4KB。

如果创建了一个cookie,并将他发送到浏览器,默认情况下它是一个会话级别的cookie(即存储在浏览器的内存中),用户退出浏览器之后即被删除。若希望浏览器将该cookie存储在磁盘上,则需要使用maxAge,并给出一个以秒为单位的时间。将最大时效设为0则是命令浏览器删除该cookie。

注意,删除cookie时,path必须一致,否则不会删除

Cookie显示商品的浏览记录

import java.io.IOException;
import java.io.PrintWriter;
import java.util.LinkedHashMap;
import java.util.Map;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

//代表首页的servlet
public class CookieDemo3 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// 1.显示网站的所有商品
response.setCharacterEncoding("UTF-8");
response.setHeader("Content-type", "text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
out.print("本网站有如下商品:<br/>");
Map<String, Book> map = Db.getAll();
// 遍历存有数据信息的“数据库”显示到页面中
for (Map.Entry<String, Book> entry : map.entrySet()) {
Book book = entry.getValue();
// 点击超链接后,在新的页面打开target="_blank"
out.print("<a href='/day07/servlet/CookieDemo4?id=" + book.getId()
+ "' target='_blank'>" + book.getName() + "</a><br/>");
}

// 2.显示用户曾经看过的商品
out.print("<br/>您曾经看过如下商品:</br>");
// 拿到请求中所有的Cookie
Cookie cookies[] = request.getCookies();
// 遍历Cookie,看看有没有浏览商品的Cookie
for (int i = 0; cookies != null && i < cookies.length; i++) {
// 如果有浏览商品的Cookie则拿到其中的商品,浏览过的商品是很多件的,商品是以,分割的
if (cookies[i].getName().equals("bookHistory")) {
// 分割字符串,拿到留有浏览过的商品的id
// 不管,有没有在正则表达式中定义,都以,分割
String[] ids = cookies[i].getValue().split("\\,"); // 2,3,1
// 遍历所有商品id获得商品
for (String id : ids) {
Book book = Db.getAll().get(id);
out.print(book.getName() + "<br />");
}
}
}
}

public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}

}

// 模拟数据库,从这个类中提取书的信息
class Db {
// 实际开发中存储数据的集合有检索数据的需求,都用双列的(map),如果没有检索数据需求,用单列的(list、set)
// LinkedHashMap是有序存储
private static Map<String, Book> map = new LinkedHashMap<String, Book>();
// 对书进行静态初始化
static {
// key是商品的id,value是商品,商品中也是有商品的id的
map.put("1", new Book("1", "javaweb开发", "老张", "一本好书!"));
map.put("2", new Book("2", "jdbc开发", "老张", "一本好书!"));
map.put("3", new Book("3", "spring开发", "老黎", "一本好书!"));
map.put("4", new Book("4", "struts开发", "老毕", "一本好书!"));
map.put("5", new Book("5", "android开发", "老黎", "一本好书!"));
}

public static Map<String, Book> getAll() {
return map;
}
}

// 模拟书类
class Book {
private String id;
private String name;
private String author;
private String description;

public Book() {
super();
}

public Book(String id, String name, String author, String description) {
super();
this.id = id;
this.name = name;
this.author = author;
this.description = description;
}

public String getId() {
return id;
}

public void setId(String id) {
this.id = id;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public String getAuthor() {
return author;
}

public void setAuthor(String author) {
this.author = author;
}

public String getDescription() {
return description;
}

public void setDescription(String description) {
this.description = description;
}
}


import java.io.IOException;
import java.io.PrintWriter;
import java.util.Arrays;
import java.util.LinkedList;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

//显示商品详细信息的servlet
public class CookieDemo4 extends HttpServlet {

public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// 1.根据用户带过来的id,显示相应商品的详细信息
response.setCharacterEncoding("UTF-8");
response.setHeader("Content-type", "text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
// 获取浏览的书的id
String id = request.getParameter("id");
// 根据id拿到书的信息
Book book = Db.getAll().get(id);
out.print(book.getId() + "<br/>");
out.print(book.getAuthor() + "<br/>");
out.print(book.getName() + "<br/>");
out.print(book.getDescription() + "<br/>");

// 2.构建Cookie,回写给浏览器
String cookieValue = buildCookie(id, request);

Cookie cookie = new Cookie("bookHistory", cookieValue);
cookie.setMaxAge(1 * 3600 * 24 * 30);
cookie.setPath("/day07");
response.addCookie(cookie);
}

// 构建Cookie的方法
private String buildCookie(String id, HttpServletRequest request) {
// 1.用户没有带Cookie过来 bookHistory=null id=1 return 1
// 2.用户带Cookie过来 bookHistory=2,5,1 id=1 bookHistory中包含id
// 要把2,5,1中的1删掉加在最前面 return1,2,5
// 3.用户带Cookie过来 bookHistory=2,5,4 id=1 假设列表最大显示3个Cookie return1,2,5
// 4.用户带Cookie过来 bookHistory=2,5 id=1 并且列表没有超出范围 return 1,2,5

// 先把要Cookie的值创建出来
String bookHistory = null;
// 得到Cookie,看看有没有bookHistory这个Cookie,有的话,把它的值赋给bookHistory(要返回的String类型的)
Cookie[] cookies = request.getCookies();
for (int i = 0; cookies != null && i < cookies.length; i++) {
if (cookies[i].getName().equals("bookHistory")) {
bookHistory = cookies[i].getValue();
}
}

// 第一种可能,那么返回的就是浏览的这个商品的id
if (bookHistory == null) {
return id;
}
// 第二种可能,要判断有没有包含,要先转成List,调用List的方法判断String.content判断出来的数据是有错误的,判断1的话21也包含
// 由于增删改查操作较多,所以用链表
LinkedList<String> list = new LinkedList(Arrays.asList(bookHistory
.split("\\,")));
/*
* if(list.contains(id)) { list.remove(id); list.addFirst(id); } else {
* //第三种和第四种可能 //第三种可能 if(list.size()>=3) { list.removeLast();
* list.addFirst(id); } else { //第四种可能 list.addFirst(id); } }
*/
// 上面的代码优化,不管怎么样都要list.addFirst(id);
if (list.contains(id)) {
list.remove(id);
} else {
if (list.size() >= 3) {
list.removeLast();
}
}
list.addFirst(id);

// 将list中的数据构建字符串返回
StringBuffer sb = new StringBuffer();
for (String bid : list) {
sb.append(bid + ",");
}
return sb.deleteCharAt(sb.length() - 1).toString();
}

public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}

}


Cookie常见应用:显示上次访问网站的时间、显示浏览过的商品、做购物(购物车)、自动登录(Cookie把用户的用户名和密码回写给浏览器,但是写这个程序的时候用户名是直接回写的,密码是通过加密后回写的)
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: