Servlet 技 术 1
复 习 几 个 要 点 2
请 求 (request) 用 户 何 时 发 起 一 个 请 求 点 击 一 个 链 接 提 交 一 个 表 单 在 地 址 栏 键 入 一 个 Web 地 址! 请 求 通 过 URL 来 指 定 http://www.neusoft.edu.cn:80/cs/news.html URL 的 构 成 : http:// 代 表 超 文 本 传 输 协 议, 通 知 服 务 器 显 示 Web 页 面 WWW Web 服 务 器 的 主 机 名 neusoft.edu.cn,web 服 务 器 所 在 的 域 名 :80, 端 口 号, 默 认 为 80 CS 服 务 器 上 的 子 目 录 news.html 文 件 夹 中 的 一 个 html 文 件! 请 求 可 以 包 含 参 数 http://localhost:8080/search.jsp?name=java+servlet&status=1 参 数 以 问 号 (?) 开 头 可 以 包 含 多 个 名 / 值 对, 之 间 用 与 号 (&) 分 隔 特 殊 字 符 必 须 进 行 编 码 ( 本 例 中 Java 和 Servlet 间 的 + 代 表 空 格 ) 3
HTTP Request 提 交 方 法 POST 在 请 求 体 中 发 送 应 用 数 据 GET 在 URL 中 发 送 应 用 数 据 HEAD 客 户 端 仅 能 获 取 响 应 的 头 部 信 息 PUT - 直 接 向 服 务 器 上 放 置 文 档 DELETE 与 PUT 执 行 相 反 的 操 作 TRACE 调 试 的 辅 助 方 法 OPTIONS 列 出 通 信 的 各 个 选 项 常 用 的 方 法 是 : GET 和 POST 4
HTTP 协 议 的 工 作 过 程 四 个 过 程 : 建 立 连 接 发 送 请 求 发 送 响 应 关 闭 连 接 建 立 连 接 : 通 过 申 请 套 接 字 (Scoket), 客 户 打 开 一 个 套 接 字, 并 将 自 己 绑 定 到 该 端 口 上 发 送 请 求 : 打 开 一 个 连 接 后, 客 户 机 将 请 求 发 送 到 服 务 器 指 定 的 端 口, 并 完 成 提 出 请 求 动 作 发 送 响 应 : 服 务 器 在 处 理 完 用 户 的 请 求 以 后, 要 想 客 户 机 发 送 相 应 信 息 关 闭 连 接 : 通 信 双 方 通 过 关 闭 套 接 字 的 办 法 关 闭 连 接 5
Servlet 简 介 Servlet 是 基 于 Servlet 框 架 ( 接 口 ) 的 java 对 象, 对 Web 服 务 器 的 功 能 进 行 扩 展, 用 来 生 成 动 态 的 内 容 简 单 地 说 Servlet 是 容 器 负 责 管 理 的 服 务 器 端 Java 程 序, 用 来 处 理 客 户 端 请 求, 并 对 客 户 端 请 求 进 行 响 应 是 SUN 公 司 提 供 的 服 务 器 端 动 态 服 务 技 术 之 一 request response Servlet EJB Web 容 器 Web 浏 览 器 6
Servlet 在 JavaEE 中 的 位 置 7
B/S 请 求 响 应 模 型 8
Servlet Architecture 客 户 端 Web 服 务 器 服 务 器 JVM 客 户 端 浏 览 器 Servlet Request Handler Servlet Engine Logs MyServlet 1. 浏 览 器 发 送 HTTP 请 求 2. Servlet 请 求 处 理 器 拦 截 请 求, 并 将 其 发 送 给 运 行 在 JVM 中 的 Servlet 引 擎 3. Servlet 引 擎 根 据 请 求 创 建 请 求 响 应 对 象 并 将 其 传 递 给 Servlet 的 services 方 法 4. Servlet 对 请 求 进 行 处 理, 并 将 输 出 输 出 给 Servlet 引 擎 5. Servlet 引 擎 将 输 出 回 送 给 Web 服 务 器 6. Web 服 务 将 请 求 回 传 给 客 户 端 浏 览 器 9
Servlet Work Flow 10
jsp 与 servlet 的 关 系 -jsp 就 是 servlet» 例 题 分 析 11
Servlet 框 架 的 对 象 模 型 <<Interface>> javax.servlet.servlet init( ) getservletconfig( ) service( ) getservletinfo( ) destroy( ) <<Interface>> javax.io.serializable javax.servlet.genericservlet init( ) getservletconfig( ) service( ) getservletinfo( ) destroy( ) getinitparameter( ) getservletcontext( ) getinitparameternames( ) getservletname( ) log( ) <<Interface>> javax.servlet.servletconfig getinitparameter( ) getservletcontext( ) getinitparameternames( ) getservletname( ) javax.servlet.http.httpservlet dodelete( ) doget( ) dooptions( ) dopost( ) doput( ) dotrace( ) getlastmodified( ) service( ) Basic Servlet 12
小 结 : Servlet 是 一 种 运 行 服 务 器 端 的 Java 应 用 程 序, 具 有 独 立 于 平 台 和 协 议 的 特 性, 并 且 可 以 动 态 地 生 成 Web 页 面, 它 工 作 在 客 户 端 请 求 与 服 务 器 响 应 的 中 间 层 Servlet 运 行 在 服 务 器 端, 与 传 统 的 以 命 令 行 方 式 启 动 的 Java 应 用 程 序 (Java Application) 不 同, Servlet 由 包 含 Java 虚 拟 机 的 Web 服 务 器 ( 如 Tomcat) 进 行 加 载 并 运 行 13
小 结 Servlet 使 用Java Servlet API 及 相 关 类 和 方 法 编 写 的 Java 应 用 程 序 除 了 Servlet API,Servlet 还 提 供 了 用 以 扩 展 基 于 HTTP 协 议 的 Java 类 软 件 包, 从 而 使 Servlet 具 有 在 Java Web 服 务 器 上 或 应 用 服 务 器 上 运 行 并 扩 展 该 服 务 器 的 能 力 Servlet API 包 括 : javax.servlet.*: 包 含 了 所 有 Servlet 类 实 现 的 基 本 接 口 和 继 承 的 基 本 类 javax.servlet.http.*: 包 含 了 编 写 基 于 HTTP 协 议 的 Servlet 所 需 的 基 类 14
创 建 第 一 个 Servlet 15
使 用 GenericServlet 处 理 请 求 Server GenericServlet subclass request response service( ) 覆 盖 service() 方 法 即 可 16
使 用 HTTPServlet 处 理 请 求 Server HttpServlet subclass Get request response doget( ) Post request service( ) response dopost( ) 覆 盖 service() 或 doxxx() 方 法 即 可, 使 用 doxxx() 方 法 时, 关 键 看 对 客 户 何 种 提 交 方 式 进 行 处 理 17
HttpServlet 类 HttpServlet 处 理 使 用 HTTP 协 议 发 送 的 请 求 和 响 应 HttpServlet 的 service() 检 查 请 求 提 交 的 方 法 并 调 用 HTTPServlet 响 应 的 方 法 : doget, dopost, doput, dodelete, dotrace, dooptions or dohead 该 类 是 一 个 抽 象 类 18
创 建 一 个 Servlet 继 承 HTTPServlet 实 现 doget 或 dopost 方 法 ( 或 两 者 都 实 现 ) 这 两 个 方 法 的 方 法 参 数 均 为 : HttpServletRequest: 用 于 获 取 Form 表 单 的 数 据 HTTP 请 求 头 部 信 息 等 等 HttpServletResponse: 用 于 设 置 HTTP 状 态 代 码 HTTP 响 应 头 部 信 息 获 取 用 于 向 客 户 端 发 送 数 据 的 输 出 流 对 象 多 数 情 况 下, 我 们 在 dopost 方 法 中 调 用 doget, 或 相 反 19
输 出 HTML 默 认 情 况 下, 响 应 中 并 没 有 设 定 内 容 的 类 型 为 了 生 成 HTML 类 型 的 文 档 : 通 过 设 置 Content-Type 头 部 信 息, 告 诉 浏 览 器 发 送 的 是 HTML 文 档 对 输 出 的 文 本 进 行 修 改 以 创 建 合 法 的 HTML 页 面 应 当 在 输 出 文 档 之 前 设 置 所 有 的 头 部 信 息 20
public class HelloWorld extends HttpServlet { public void doget(httpservletrequest request, HttpServletResponse response) throws ServletException, IOException {! PrintWriter out = response.getwriter(); HelloWorld.java out.println("<html><head><title>hello World</title></head>\n"); out.println("<body>"); out.println("<h2>" + new java.util.date() + "</h2>\n"); out.println("<h1>hello World</h1>\n</body></html>"); } } 21
配 置 Servlet 器 <web-app> web.xml <servlet> <servlet-name>hello</servlet-name> <servlet-class>helloworld</servlet-class> </servlet> <servlet-mapping> myapp/web-inf/classes/helloworld.class <servlet-name>hello</servlet-name> <url-pattern>/hello</url-pattern> </servlet-mapping></web-app> </web-app> 22
Servlet 的 生 命 周 期 23
Servlet 生 命 周 期 24
Servlet 生 命 周 期 第 一 次 请 求 Servlet 1 2 3 init() 5 4 service() 再 次 请 求 Servlet 1 2 3 service() 4 25
Servlet 生 命 周 期 调 用 Init 方 法 进 行 初 始 化 设 置 创 建 Servlet 实 例 调 用 service 方 法 处 理 客 户 端 请 求 ServletConfig 调 用 destroy 方 法 销 毁 Servlet 实 例 并 释 放 占 用 资 源 加 载 Servlet 类 垃 圾 回 收 26
从 请 求 中 获 取 信 息 27
获 取 HTTP 协 议 相 关 的 数 据 可 以 通 过 HttpServletRequest 对 象 获 取 HTTP 请 求 的 值 获 取 头 信 息 hdr 可 以 使 用 请 求 参 数 的 getheader( hdr ) 方 法 进 行 获 取 获 取 所 有 的 头 部 信 息 名 可 以 使 用 方 法 : getheadernames() 获 取 特 定 请 求 信 息 的 方 法 有 : getcookies, getcontentlength, getcontenttype, getmethod, getprotocol, 等 28
通 用 HTTP 头 部 信 息 29
请 求 头 部 信 息 30
获 取 Form 表 单 中 的 数 据 使 用 HTML 表 单, 我 们 可 以 向 Web 应 用 发 送 参 数 <form action= method= > </form> 组 成 了 一 个 简 单 的 Form 表 单 : action: 表 单 数 据 要 发 送 到 的 Web 应 用 的 地 址 method: 发 送 数 据 是 所 使 用 HTTP 协 议 的 提 交 方 法 ( 也 就 是 get 或 post) 31
<input> <select> <textarea> 标 记 在 表 单 内 部, INPUT Textera Select 定 义 了 用 于 输 入 数 据 的 各 种 输 入 框 标 准 的 输 入 类 型 包 括 : 按 钮 (button), 复 选 框 (checkboxe), 密 码 框 (password ), 单 选 按 钮 (radio ), 单 行 文 本 (text), 图 片 (image ), 多 行 文 本 编 辑 器 (text area), 隐 藏 字 段 (hidden), 等 等. 它 们 都 是 以 String 类 型 的 值 同 指 定 的 参 数 相 关 联 32
GET 示 例 <form method="get" action="http://www.google.com/search"> <p><input name="q" type="text" /> <input type="submit" /> <input type="reset" /> </p> </form>! http://www.google.com/search?q=servlets 33
POST 示 例 <form method="post" action="http://www.google.com/search"> <p><input name="q" type="text" /> <input type="submit" /> <input type="reset" /> </p> </form> Google 并 不 支 持 Post 方 法! 34
获 取 Parameter 值 设 置 请 求 编 码 格 式 setcharacterencoding( code ) 要 获 取 名 为 x 的 参 数 值 : req.getparameter("x") 这 里 req 是 Servlet 的 请 求 参 数 如 果 该 参 数 的 值 可 能 有 多 个 则 使 用 : req.getparametervalues("x") 要 想 获 取 所 有 参 数 名 : req.getparameternames() 35
<html><head><title>sending Parameters</title> <style type="text/css"> p{display:table-row} span{display:table-cell; padding:0.2em} </style></head><body>! <h1>please enter the parameters</h1> <form action="setcolors" method="get"> <p>background color: <span><input type="text" name="bgcolor"/></span></p> <p>font color: <span><input type="text" name="fgcolor"/> </span> </p> <p>font size: <span><input type="text" name="size"/></span></p> <h2> <input type="submit" value="submit Parameters"/></h2> </form>! </body></html> 36 parameters.html
示 例 (cont) public class SetColors extends HttpServlet { public void doget(httpservletrequest request,httpservletresponse response) throws ServletException, IOException {! response.setcontenttype("text/html"); PrintWriter out = response.getwriter(); String bg = request.getparameter("bgcolor"); String fg = request.getparameter("fgcolor"); String size = request.getparameter("size"); SetColors.java 37
示 例 (cont)!! } out.println("<html><head><title>set Colors Example" +"</title></head>"); out.println("<body style=\"color:" + fg + ";background-color:" + bg + ";font-size:"+ size + "px\">"); out.println("<h1>set Colors Example</h1>"); out.println("<p>you requested a background color " + bg + "</p>"); out.println("<p>you requested a font color " + fg + "</p>"); out.println("<p>you requested a font size " + size + "</p>"); out.println("</body></html>"); SetColors.java 38
处 理 Post 提 交 的 数 据 获 取 POST 方 法 提 交 的 数 据 的 方 式 与 GET 方 法 相 同!! 因 此 可 以 直 接 在 dopost 方 法 中 调 用 doget 方 法 <form action="http://www.edu.neusoft.cn/ HelloWorldExample" method="post"> public void dopost(httpservletrequest request, HttpServletResponse response) throws ServletException, IOException {! doget(request, response); } 39
Servlet 的 会 话 管 理 40
为 什 么 要 进 行 会 话 管 理 HTTP 是 一 个 无 状 态 的 通 信 协 议 (Stateless protocol) 造 成 的 结 果 是 为 了 解 决 这 个 问 题 提 供 了 : Cookie Session 41
会 话 的 建 立 过 程 id 1 Web 浏 览 器 1 request 设 置 cookie id 1 response id 1 Web 服 务 器 request response 创 建 session! Servlet 42
会 话 的 建 立 过 程 id 2 Web 浏 览 器 2 request 设 置 cookie id 2 id 2 response id 1 Web 服 务 器 request response 创 建 Session! Servlet 43
会 话 的 建 立 过 程 id 1 Web 浏 览 器 1 request Cookie: id 1 id 2 response id 1 Web 服 务 器 request response Session 读 / 写! Servlet 44
会 话 的 建 立 过 程 id 2 Web 浏 览 器 2 request Cookie: id 2 id 2 response id 1 Web 服 务 器 request response Session 读 / 写 Servlet Servlet 45
访 问 Session 数 据 HttpSession 表 示session 对 象 在 doxxx 方 法 中 调 用 请 求 对 象 的 getsesssion() or getsession(true) 获 取 或 创 建 session 对 象 ( 如 果 session 对 象 不 存 在 ) Session 创 建 时 服 务 器 会 自 动 向 响 应 对 象 中 添 加 一 个 用 于 建 立 会 话 目 的 的 cookie 若 session 不 存 在, 且 不 想 创 建 新 的 session 时 调 用 getsession(false) 46
HttpSession 的 方 法 Session 中 的 数 据 是 通 过 哈 希 表 的 方 式 进 行 存 取 的 : - setattribute(string name,object value) - Object getattribute(string name) 更 多 的 方 法 : - removeattribute, getattributenames - isnew, invalidate, getid - getcreationtime, getlastaccessedtime - getmaxinactiveinterval, setmaxinactiveinterval 47
页 面 间 数 据 共 享 方 式 重 写 URL 在 HTTP 协 议 中,Get 方 式 可 以 实 现 数 据 由 客 户 端 到 服 务 器 端 的 传 送 而 重 写 URL 正 是 利 用Get 的 函 数, 可 以 URL 的 后 面 附 加 参 数, 和 服 务 器 的 请 求 一 起 发 送, 这 些 参 数 称 为 查 询 字 符 串 查 询 字 符 串 为 一 对 名 称 / 值 对,URL 中 多 个 查 询 字 符 串 之 间 使 用 & 符 号 连 接! 示 例 : http://localhost:8080/ch7/pages/newsdetail.jsp? id=10&action=delete! String id = request.getparameter("id"); String action = request.getparameter("action"); 48
页 面 间 数 据 共 享 方 式 重 写 URL 的 缺 点 :! 重 写 URL 的 页 面 必 须 动 态 生 成 ; 用 户 数 据 暴 露 在 URL 上, 造 成 安 全 上 的 隐 患 ; URL 的 长 度 有 限 制, 如 使 用URL 传 递 大 量 数 据, 会 造 成 性 能 下 降 ; 必 须 对 所 有 指 向 本 Web 站 点 的 URL 进 行 编 码 ; 不 能 预 先 记 录 下 页 面 的 URL, 或 者 从 其 他 的 Web 站 点 链 接 访 问 49
页 面 间 数 据 共 享 方 式 共 享 会 话 使 用 重 写 URL 共 享 用 户 数 据 时, 还 有 一 个 限 制 就 是 两 个 页 面 间 必 须 通 过 链 接 关 联 起 来 如 果 两 个 链 接 这 种 关 联 关 系, 将 无 法 使 用 重 写 URL 共 享 数 据 那 么 可 以 使 用 内 置 对 象 session, 即 共 享 会 话 的 形 式 共 享 用 户 数 据 session 对 象 的 setattribute(string name, Object obj) 方 法 可 以 设 置 在 会 话 期 内 共 享 的 数 据 属 性 名 称 及 值 在 同 一 个 用 户 的 同 一 个 会 话 期 内, 不 同 的 数 据 属 性 名 称 要 区 别, 若 重 名 时 原 有 的 属 性 值 将 被 替 换 使 用getAttribute(String name) 可 以 从 会 话 中 检 索 对 应 属 性 名 称 的 值, 需 要 注 意 的 是 该 方 法 的 返 回 值 为 Object 类 型, 因 此 常 常 需 要 数 据 类 型 转 换 50
页 面 间 数 据 共 享 方 式 共 享 会 话 示 例 : HttpSession session = request.getsession(true); // 设 置 会 话 有 效 期 为 20 分 钟 session.setmaxinactiveinterval(1200); // 向 会 话 中 添 加 一 个 属 性 名 称 为 users, 值 为 regusers 的 对 象 session.setattribute("users", regusers); // 获 取 属 性 名 称 为 users 的 对 象 值, 进 行 强 制 类 型 转 换 users u = (users)session.getattribute("users"); // 从 会 话 中 删 除 属 性 名 称 为 users 的 对 象 session.removeattribute("users"); 51
页 面 间 数 据 共 享 方 式 在 web.xml 配 置 会 话 有 效 期 为 30 分 钟 <session-config> <session-timeout> 30 </session-timeout> </session-config> 52
页 面 间 数 据 共 享 方 式 Cookie 用 于 存 储 Web 服 务 器 发 送 给 客 户 端 的 信 息 ( 通 常 以 文 本 文 件 的 形 式 保 存 在 客 户 机 的 硬 盘 上 ) 当 客 户 端 第 一 次 访 问 服 务 器 时, 服 务 器 会 为 客 户 创 建 一 个 Cookie 对 象, 在 响 应 客 户 端 的 同 时 把 Cookie 对 象 发 送 到 客 户 端, 在 以 后 的 访 问 中, 客 户 端 会 在 请 求 的 同 时 将 Cookie 对 象 同 时 发 送 到 服 务 器 因 此, 在 实 际 应 用 中 可 以 把 需 要 共 享 的 数 据 保 存 至 Cookie 中, 需 要 这 些 数 据 时 再 从 客 户 端 的 Cookie 中 读 出, 从 而 实 现 共 享 数 据 53
页 面 间 数 据 共 享 方 式 Cookie 也 是 以 属 性 名 / 值 对 的 形 式 保 存 数 据 的, 这 些 信 息 可 被 封 装 在 Cookie 对 象 中! 当 客 户 端 第 一 次 访 问 Web 服 务 器 时, 首 先 要 生 成 一 组 Cookie 信 息 并 保 存 在 客 户 端 中 在 Java Web 应 用 开 发 中, 是 通 过 内 置 对 象 response 的 addcookie() 方 法 来 实 现 的 第 一 步 : 通 过 Cookie 构 造 方 法 创 建 Cookie 对 象 Cookie cookie = new Cookie(String key,object value);! 第 二 步 : 设 置 Cookie 的 最 大 保 留 时 间 cookie.setmaxage(int value);// 单 位 为 秒! 第 三 步 : 将 Cookie 对 象 添 加 到 响 应 对 象 中 responsee.addcookie(cookie obj);! 第 四 步 : 得 到 客 户 端 发 送 过 来 的 Cookie 对 象 Cookie [] cookies = request.getcookies();// 此 方 法 将 返 回 一 个 Cookie 数 组! 第 五 步 :Cookie 中 的 方 法 cookie.getname();// 得 到 Cookie 中 的 属 性 名 称, cookie.getvalue();// 得 到 Cookie 中 的 值 54
Cookie HTTP 协 议 是 一 种 无 连 接 的 协 议 ( 主 要 说 明 为 什 么 需 要 Cookie 和 session) Cookie 是 Web 服 务 器 保 存 在 用 户 硬 盘 上 的 一 段 文 本 Cookie 允 许 一 个 Web 站 点 在 用 户 的 电 脑 上 保 存 信 息 并 且 随 后 再 取 回 它 信 息 的 片 断 以 名 / 值 对 (name-value pairs) 的 形 式 储 存 在 某 种 程 度 上,cookie 和 session 内 置 对 象 有 些 相 似, 但 二 者 存 在 本 质 的 差 别 : Cookie 存 在 与 客 户 端,session 存 在 与 服 务 器 Cookie 主 要 用 户 保 存 脱 机 数 据, 而 session 内 置 对 象 主 要 用 于 跟 踪 用 户 会 话 但 他 们 也 存 在 一 些 共 同 点, 那 就 是 不 论 是 cookie 还 是 session 内 置 对 象 都 需 要 浏 览 器 支 持 cookie 并 且 没 有 禁 用 cookie 55
Cookie 作 用 :Cookie 对 象 通 常 用 于 在 浏 览 器 端 保 存 会 话 过 程 中 的 一 些 参 数 当 浏 览 器 请 求 服 务 器 的 页 面 时 自 动 发 送 到 服 务 器 上 Cookie 是 文 本 文 件, 保 存 路 径 C:\Documents and Settings\ 登 录 用 户 名 \Cookies Cookie 的 常 用 用 途 : Cookie 是 站 点 跟 踪 特 定 访 问 者 访 问 的 次 数, 最 后 访 问 的 时 间 以 及 访 问 者 进 入 站 点 的 路 径 ; Cookie 能 够 帮 助 站 点 统 计 用 户 个 人 资 料 以 实 现 各 种 各 样 的 个 性 化 服 务 cookie 实 现 自 动 登 录 功 能, 使 得 用 户 不 需 要 输 入 用 户 名 和 密 码 就 可 以 进 入 曾 经 浏 览 的 站 点 56
Cookie 使 用 的 限 制 每 个 客 户 端 最 多 存 放 300 个 Cookie 每 个 Cookie 的 数 据 最 大 为 4KB 每 个 服 务 器 或 域 最 多 向 同 一 客 户 端 写 20 个 cookie 57
Servlets 与 Cookies Servlet API 提 供 了 处 理 cookie 的 机 制 类 javax.servlet.http.cookie 表 示 cookie 获 取 方 法 : getname(), getvalue(), getpath(), getdomain(), getmaxage(), getsecure() 设 置 方 法 : setvalue(), setpath(), setdomain(), setmaxage() 58
Servlets 与 Cookies ( 续 ) 从 请 求 中 获 取 Cookie: Cookie[] HttpServletRequest.getCookies() 向 响 应 中 添 加 cookie: HttpServletResponse.addCookie(Cookie cookie) 59
向 客 户 端 写 入 Cookie 的 步 骤 创 建 Cookie Cookie c=new Cookie( name, value ); 设 置 Cookie 的 属 性 c.setmaxage(60*60*24); c.setdomain( *.neusoft.edu.cn ) c.setpath( / ) 调 用 response 的 addcookie 方 法 将 其 写 入 到 客 户 端 response.addcookie(c) 60
读 取 Cookie 的 步 骤 从 客 户 端 读 取 所 有 能 够 被 本 服 务 器 读 取 的 Cookie Cookie[] cookies=request.getcookies(); 从 中 找 出 需 要 的 Cookie for(int i=0;i<cookies.length;i++){ If(cookies[i].getName().equals( wanted )){ Do something useful } } 61
示 例 <html> getname.html <head><title>insert your Name</title></head> <body> <h1>what is your name?</h1> <form action="welcomeback" method="get"> <p> <input type="text" name="username" /> <input type="submit" /> </p> </form> </body> </html> 62
! 示 例 : WelcomeBack.java public class WelcomeBack extends HttpServlet { public void doget(httpservletrequest req, HttpServletResponse res) throws ServletException, IOException { String user = req.getparameter("username"); if (user == null) { // Find the "username" cookie Cookie[] cookies = req.getcookies(); for (int i = 0; cookies!= null && i < cookies.length; ++i) { if (cookies[i].getname().equals("username")) user = cookies[i].getvalue(); } } else res.addcookie(new Cookie("username", user)); 63
示 例 (cont)!! if (user == null) // No parameter and no cookie res.sendredirect("getname.html"); res.setcontenttype("text/html"); PrintWriter out = res.getwriter(); out.println("<html><body><h1>welcome Back " + user + "</h1></body></html>"); } } WelcomeBack.java 64
请 求 转 发 与 重 定 向 Internet 的 一 个 主 要 特 征 就 是 通 过 超 级 链 接 聚 合 了 大 量 的 信 息 资 源,Web 应 用 中 各 个 资 源 ( 包 括 HTML/ JSP 页 面 Servlet 对 象 等 ) 也 可 以 实 现 资 源 的 相 互 关 联 与 整 合, 在 Java Web 应 用 中 可 使 用 以 下 几 种 方 式 达 到 Web 资 源 的 关 联 与 整 合 : 请 求 转 发 ; 重 定 向 ; 包 含 65
请 求 转 发 与 重 定 向 请 求 转 发 RequestDispatcher Servlet 66
请 求 转 发 与 重 定 向 // 转 发 至ForwardPage.jsp 页 面 RequestDispatcher dispatcher = request.getrequestdispatcher("forwardpage.jsp"; dispatcher.forward(request, response);! RequestDispatcher 对 象 可 以 把 用 户 对 当 前 Web 资 源 的 请 求 转 发 给 其 他 Web 资 源, 如 上 述 代 码 中 转 发 给 ForwardPage.jsp 页 面, 而 且 将 用 户 对 当 前 Servlet 的 请 求 和 响 应 (HttpServletRequest 对 象 实 例 request 和 HttpServletResponse 对 象 实 例 response) 传 递 给 ForwardPage.jsp 67
请 求 转 发 与 重 定 向 注 意 : 请 求 转 发 方 式 通 常 用 于 Servlet 中, 而 在 JSP 页 面 中 可 以 使 用<jsp:forward> 动 作 标 记 达 到 相 同 的 效 果 另 外, 请 求 转 发 方 式 只 能 把 请 求 转 发 给 同 一 个 Web 应 用 中 的 资 源, 并 不 能 转 发 给 其 他 Web 应 用 中 的 资 源 ; 并 且 使 用 请 求 转 发 的 Web 资 源 与 目 标 Web 资 源 之 间 共 享 request 范 围 内 的 数 据 68
请 求 转 发 与 重 定 向 重 定 向 是 将 来 自 客 户 端 的 请 求 传 递 给 其 他 的 Web 资 源, 与 请 求 转 发 不 同 的 是, 目 标 资 源 可 以 是 不 在 同 一 个 Web 应 用 中 在 Servlet 中, 使 用HttpServletResponse 对 象 的 sendredirect() 方 法 重 定 向 response.sendredirect("pages/confirmregister.jsp"); 与 请 求 转 发 相 比, 重 定 向 与 请 求 转 发 的 区 别 在 于 : Web 资 源 可 以 重 定 向 任 何 一 个 URL, 而 不 仅 仅 局 限 于 同 一 个 Web 应 用 中 重 定 向 是 返 回 一 个 应 答 给 客 户 端, 然 后 再 重 新 发 送 一 个 请 求 给 目 标 URL, 所 以 浏 览 器 地 址 栏 会 更 新 为 目 标 Web 资 源 的 URL; 而 使 用RequestDispatcher 对 象 请 求 转 向 时, 浏 览 器 地 址 栏 不 会 更 新, 仍 为 原 有 Web 资 源 的 URL 使 用 重 定 向 时,Web 资 源 之 间 不 会 共 享 使 用 用 户 请 求 对 象, 而 RequestDispatcher 则 共 享 69
请 求 转 发 器 RequestDispatcher 对 象 用 来 将 客 户 发 送 的 请 求 发 送 给 服 务 器 的 其 他 资 源 该 资 源 可 以 是 静 态 资 源 ( 如 HTML 文 件 ) 也 可 以 动 态 资 源 ( 如 Servlet 或 JSP 文 件 ) 要 将 请 求 发 送 给 资 源 x, 使 用 : getservletcontext().getrequestdispatcher( /x") request.getrequestdispatcher("x") 70
RequestDispatcher 提 供 的 方 法 void forward(servletrequest request, ServletResponse response) 将 请 求 从 当 前 的 Servlet 转 发 到 其 他 资 源 void include(servletrequest request, ServletResponse response) 在 响 应 对 象 中 包 含 资 源 的 内 容 71
数 据 的 传 递 属 性 传 递 在 对 Servlet 或 JSP 进 行 转 发 时 有 三 种 不 同 的 传 递 参 数 的 方 式 仅 在 请 求 范 围 内 使 用 该 数 据 : request.setattribute("key", value); 客 户 在 会 话 过 程 中 使 用 的 数 据 : session.setattribute("key", value); 任 何 用 户 都 可 以 访 问 的 数 据 context.setattribute("key", value); 参 数 传 递 通 过 在 地 址 栏 后 面 追 加 参 数 url?parameter1=value1¶meter2=value2 72
示 例 在 JokesAndImages Servlet, 用 户 可 以 随 机 选 择 一 个 随 机 的 笑 话 或 图 片 在 服 务 器 的 images/ 目 录 下 有 五 个 图 片, 在 jokes/ 下 有 五 个 笑 话 (txt 文 件 ) 空 请 求 会 被 转 发 给 HTML 文 件, 使 得 用 户 可 以 选 择 笑 话 或 图 片 选 择 笑 话, 请 求 就 会 转 发 给 JokesServlet 选 择 图 片, 请 求 就 会 随 机 转 发 给 images/ 目 录 下 的 某 个 图 片 73
Jokes and Images <html> <head><title>images and Jokes</title></head> <body> <h1>please Select:</h1> <form method="post" action="jokesandimages"> <h2> <input type="submit" name="joke" value="a Joke" /> <input type="submit" name="image" value="an Image" /> </h2> </form> </body></html> imagesjokesoptions.html 74
public class JokesAndImages extends HttpServlet { public void dopost(httpservletrequest req, HttpServletResponse res) throws ServletException, IOException { } int randomnum = 1 + Math.abs((new Random()).nextInt() % 5); if (req.getparameter("joke")!= null) { req.setattribute("jokenumber", new Integer(randomNum)); getservletcontext().getrequestdispatcher("/ Jokes").forward(req,res); } else if (req.getparameter("image")!= null) { getservletcontext().getrequestdispatcher("/images/image" + randomnum + ".gif").forward(req, res); } else getservletcontext().getrequestdispatcher ("/imagesjokesoptions.html"). forward(req,res); public void doget... }} JokesAndImages.java 75
Jokes and Images ( 续 ) public class Jokes extends HttpServlet { public void dopost(httpservletrequest req, HttpServletResponse res) throws ServletException, IOException { res.setcontenttype("text/html"); PrintWriter out = res.getwriter(); out.println("<html><body><h1>a Joke</h1><pre>"); int jokenum = ((Integer) req.getattribute("jokenumber")).intvalue(); getservletcontext().getrequestdispatcher ("/jokes/joke" + jokenum + ".txt").include(req, res); out.println("\n</pre>"); out.println("<a href=\"" + req.getrequesturl() + "\">Back</a>"); out.println("</body></html>"); }} Jokes.java 76
转 发 vs 重 定 向 默 认 情 况 下, SendRedirect 不 会 保 留 请 求 对 象 中 参 数 SendRedirect 会 在 客 户 端 呈 现 跳 转 后 的 URL 地 址 77
ServletContext 78
ServletContext 对 象 servlet 容 器 在 启 动 时 会 加 载 web 应 用, 并 为 每 个 web 应 用 创 建 唯 一 的 servlet context 对 象, 可 以 把 ServletContext 看 成 是 一 个 Web 应 用 的 服 务 器 端 组 件 的 共 享 内 存, 在 ServletContext 中 可 以 存 放 能 够 被 所 有 用 户 访 问 的 共 享 数 据 ServletContext 对 象 只 在 web 应 用 被 关 闭 的 时 候 才 被 销 毁, 不 同 的 web 应 用, ServletContext 各 自 独 立 存 在 79
ServletContext 对 象 每 个 Web 应 用 只 有 一 个 ServletContext 对 象 可 以 使 用 getservletcontext() 方 法 获 取 ServletContext 对 象 ServletContext 拥 有 众 多 的 方 法 例 如, 你 可 以 在 该 对 象 中 存 储 信 息, 注 意 存 储 在 此 处 的 数 据 会 伴 随 Web 应 用 的 整 个 生 命 周 期, 除 非 明 确 地 将 其 移 除 80
示 例 : 计 数 服 务 public class CounterServlet extends HttpServlet { public void init() throws ServletException { Integer counter = (Integer)getServletContext().getAttribute("counter"); } } if(counter == null) { getservletcontext().setattribute("counter",new Integer(0)); 81
public void doget(httpservletrequest req,httpservletresponse res) throws ServletException, IOException {! PrintWriter out = res.getwriter(); res.setcontenttype("text/html"); int counter = 0; synchronized(this) { counter = ((Integer)getServletContext(). getattribute("counter")).intvalue(); getservletcontext(). setattribute("counter",new Integer(++counter)); } out.println("<html><head><title>counter</title><body><h1>" + "[" + counter + "]</h1></body></html>"); }} 82
使 用 ServletContext 为 了 同 Servlet 容 器 ( 如 Tomcat) 进 行 通 信, 需 要 使 用 ServletContext 对 象 Web 应 用 的 所 有 Servlet 都 共 用 着 同 一 个 ServletContext 可 以 存 储 Web 应 用 的 初 始 参 数 可 以 存 储 和 操 纵 应 用 范 围 内 容 共 享 的 属 性 可 以 访 问 日 志 记 录 对 象 可 以 将 请 求 分 发 到 其 他 资 源 83
ServletContext 提 供 的 方 法 访 问 初 始 参 数 的 方 法 : getinitparameter(string name), getinitparameternames() 读 取 web 应 用 属 性 的 方 法 : getattribute(string name), getattributenames() 操 纵 web 应 用 属 性 的 方 法 : setattribute(string, Object), removeattribute(string) 将 相 对 路 径 转 换 为 绝 对 路 径 的 方 法 : getrealpath(string path), URL getresource(string path) 84
ServletContext 提 供 的 方 法 输 出 web 应 用 的 日 志 : log(string msg), log(string message, Throwable exception) 获 取 资 源 转 发 器 ( 稍 后 讨 论 ): RequestDispatcher getrequestdispatcher(string path) 获 取 Servlet 容 器 的 名 称 和 版 本 : String getserverinfo() 85
有 关 ServletContext 的 注 意 事 项 每 个 Web 应 用 只 有 一 个 ServletContext 对 象 同 一 个 Web 应 用 中 的 不 同 Servlet, 调 用 getservletcontext 方 法 获 得 的 是 同 一 个 ServletContext 对 象 示 例 可 以 对 其 锁 定 进 行 锁 定, 用 以 防 止 web 应 用 程 序 对 关 键 部 分 的 访 问 86
FILTER Filter 也 称 之 为 过 滤 器, 它 是 Servlet 技 术 中 非 常 重 要 的 技 术 之 一,Web 开 发 人 员 通 过 使 用Filter 技 术, 可 管 理 Web 服 务 器 上 所 有 Web 资 源, 如 JSP Servlet 静 态 页 面 等 进 行 拦 截, 从 而 实 现 一 些 特 殊 的 功 能 例 如 实 现 URL 级 别 的 权 限 访 问 控 制 过 滤 敏 感 词 汇 字 符 编 码 转 换 等 一 些 高 级 应 用 87
FILTER Filter 能 够 在 一 个 请 求 到 达 Servlet 之 前 预 处 理 用 户 请 求, 也 可 以 在 离 开 Servlet 时 处 理 HTTP 响 应 归 纳 起 来,Filter 可 以 完 成 如 下 工 作 在 执 行Servlet 之 间, 首 先 执 行Filter 程 序, 并 为 之 做 一 些 预 处 理 工 作 ; 根 据 程 序 需 要 修 改 请 求 和 响 应 ; 在 Servlet 被 调 用 之 后 截 获 Servlet 的 执 行 88
FILTER Filter API 主 要 包 括 3 个 主 要 接 口: Filter 接 口 FilterChain 接 口 和 FilterConfig 接 口 这 些 接 口 都 存 放 在 javax.servlet 包 中! Filter 接 口 包 含 以 下 方 法 :init() destroy() 和 dofilter() 方 法 ;dofilter(servletrequest request, ServletResponse response,filterchain chain) 方 法 是 Filter 的 核 心 方 法,Filter 可 以 通 过 该 方 法 对 请 求 和 响 应 进 行 处 理, 如 通 过 调 用 该 方 法 收 集 数 据 过 滤 器 通 过 传 送 给 此 方 法 的 FilterChain 参 数, 以 此 调 用 chain.dofilter() 将 控 制 权 传 送 给 下 一 个 过 滤 器 89
FILTER FilterChain 接 口 中 也 有 一 个 dofilter() 方 法, 即 void dofilter(servletrequest request, ServletResponse response) 此 方 法 是 由 Servlet 容 器 提 供 给 开 发 者 的, 用 于 对 资 源 请 求 过 滤 链 的 依 次 调 用, 通 过 FilterChain 调 用 过 滤 链 中 的 下 一 个 过 滤 器, 如 果 是 最 后 一 个 过 滤 器, 则 下 一 个 就 调 用 目 标 资 源 90
FILTER FilterConfig 接 口 可 获 取 过 滤 器 名 初 始 化 参 数 以 及 活 动 的 Servlet 上 下 文, 该 接 口 提 供 了 以 下 4 个 方 法 String getfiltername(): 返 回 web.xml 文 件 中 定 义 的 该 过 滤 器 的 名 称 ; ServletContext getservletcontext(): 返 回 调 用 者 所 处 的 Servlet 上 下 文; String getlnitparameter(string name): 返 回 过 滤 器 初 始 化 参 数 值 的 字 符 串 形 式, 当 参 数 不 存 在 时, 返 回 nul1; public Enumeration getinitparameternames(): 以 Enumeration 形 式 返 回 过 滤 器 所 有 初 始 化 参 数 值, 如 果 没 有 初 始 化 参 数, 返 回 为 null 91
LISTENER Listener 又 称 为 监 听 器, 从 字 面 上 可 以 看 出 Listener 主 要 用 来 监 听 之 用 通 过 Listener 可 以 监 听 Web 服 务 器 中 某 一 个 执 行 动 作, 并 根 据 其 要 求 做 出 相 应 的 响 应 92
LISTENER Servlet 2.4 规 范 中 提 供 了 8 个 Listener 接 口, 可 以 将 其 分 为 三 类, 分 别 如 下 第 一 类 : 与 ServletContext 有 关 的 Listener 接 口 包 括 两 个 Listener 接 口, 分 别 是 ServletContextListener 和 ServletContextAttributeListener 第 二 类 : 与 HttpSession 有 关 的 Listener 接 口 包 括 四 个 Listener 接 口, 分 别 是 HttpSessionListener HttpSessionAttributeListener HttpSessionBindingListener 和 HttpSessionActivationListener 第 三 类 : 与 ServletRequest 有 关 的 Listener 接 口 包 括 两 个 Listener 接 口, 分 别 是 ServletRequestListener 和 ServletRequestAttributeListener 93
MVC 模 式 简 介 MVC 即 Model-View-Controller( 模 型 - 视 图 - 控 制 器 ) 是 一 种 软 件 设 计 模 式,MVC 最 早 出 现 在 Smalltalk 语 言 中, 后 来 在 Java 中 得 到 广 泛 应 用, 并 且 被 Sun 公 司 推 荐 为 Java EE 平 台 的 设 计 模 式, 后 来 Sun 公 司 提 出 了 两 个 基 于 MVC 模 式 的 Java Web 开 发 模 型 :JSP Model1 和 JSP Model2 94
MVC 模 式 简 介 MVC 模 块 间 的 协 作 关 系 : 客 户 通 过 应 用 程 序 发 出 请 求, 该 请 求 转 发 给 控 制 器 ; 控 制 器 接 受 用 户 请 求, 并 决 定 使 用 何 种 业 务 逻 辑 处 理 该 请 求, 并 调 用 相 应 的 模 型 处 理 ; 模 型 处 理 用 户 请 求 并 存 取 相 关 数 据 表 示 客 户 查 询 检 索 出 的 任 何 数 据 都 被 返 回 给 控 制 器 ; 控 制 器 接 收 从 模 型 返 回 的 数 据, 并 选 择 适 当 的 视 图 显 示 响 应 结 果 95
MVC 模 式 简 介 Java EE 应 用 的 架 构 发 展 过 程 - 传 统 方 式 JSP 96
MVC 模 式 简 介 Java EE 应 用 的 架 构 发 展 过 程 -JSP Model 1 JSP JavaBean 97 Web
. MVC 模 式 简 介 Java EE 应 用 的 架 构 发 展 过 程 -JSP Model 2 Servlet JSP JavaBean Web 98
JSP 中 实 现 MVC 模 式 例 子 实 现 思 想 : 利 用MVC 模 式 的 思 想 实 现 用 户 注 册 功 能 ; 设 计 用 户 注 册 的 JSP 页 面; 设 计 JavaBean; 设 计 Servlet; 整 合 应 用! 代 码 99