参考韩顺平Servlet教学视频:学习笔记自行整理
MVC模式修改用户管理系统
对当前网站结构的问题分析
1. 在LoginCl.java 文件和ManageUser.java文件中都去操作了数据库,它们的逻辑相似,有重复代码
2: 整个框架没有清晰的层次关系,显得非常乱.
3:代码一点也不优雅, 可读性差,可维护性差.
解决方法:
指导思想:
① 业务逻辑代码和界面分离
② 把常用的代码(对数据库的连接和操作) 封装到工具类
具体的方法
① 每一张表对应 一个 domain类(表示数据) 还要对应一个 Service 类
比如 users 表 对应 Users 类(domain 类) UserService类(该类会封装对users表的各种操作) ,实际上这里体现出 数据 和 操作分离的思想
上机练习:
包登录部分改成mvc模式 ,建议先保存一份项目,再改.
① 完成分页的mvc模式改写
首先在UsersService 类中添加方法
//? 为什么要返回ArrayList ,而不是我们想到 ResultSet
//1. ArrayList 中封装 User对象,更加符合面向对象的编程方式 OOP
//2.我们通过 Resulst->User 对象->ArrayList 这样ArrayList 和 Resultset没有关系,就可以及时关闭数据库资源.
public ArrayList getUsresByPage(int pageNow,int pageSize){
reutrn al;
}
练习: 把我们的用户管理系统出了 cookie 和 session 相关的功能不做,其它都做了.
一起完成用户管理系统的crud操作
1. 一个请求对应一个控制器
优点: 逻辑清晰
缺点: 会造成控制器过多
可以这样考虑:一类事务请求,我们做一个控制器,即让该控制器可以处理多个请求,为了让一个控制器去区分,不同的请求,我们可以这样做:
在发出请求的同时,在带一个type=del 或者 type=add 或者 type=update..., 在控制器中我们接收type的值,从而判断用户希望做什么事情!
关于跳转到修改用户界面有两种思路:
(1) 传递用户id号的同时,把用户的其它信息一并传递,这样可以减少数据库查询的次数 (缺点: 增加网络开销 100字节*1000000*2 , 优点: 减少对数据库的一次操作)
(2) 只传递用户id好,控制器接收到id后,再查询数据库,从而显示.
添加用户
(1) 用户id号,我们做成子增长
在oracle中先建立一个sequece
create sequence users_seq
start with 11
increment by 1
minvalue 11
nomaxvalue
nocycle
nocache
课堂练习
添加查询用户功能:安装老师给出网站结构写.
会话技术cookie和session
什么是会话
基本概念: 指用户开一个浏览器,访问一个网站,只要不关闭该浏览器,不管该用户点击多少个超链接,访问多少资源,直到用户关闭浏览器,整个这个过程我们称为一次会话.
比如打电话.
为什么需要cookie技术(会话技术)
如何保存用户上次登录时间
如何显示用户浏览历史?
如何把登录的用户名和密码电脑,下次登录,不需要重新输入:
解决之道—cookie
cookie的原理图
cookie的小结
① cookie 是在服务端创建
② cookie 是保存在浏览器这端
③ cookie 的生命周期可以通过
cookie.setMaxAge(2000);
☞ 如果不设置setMaxAge则该cookie的生命周期当浏览器关闭时,就消亡.
④ cookie 可以被多个浏览器共享(与session的区别)
⑤ 怎么理解
我们可以把cookie 想成一张表
?如果cookie重名会有什么问题?
如果重名就会替换存在的cookie值.
⑥ 一个web应用可以保存多个cookie,但保存在同一个cookie文本在客户端浏览器下
⑦ cookie存放的时候是以明文方式存放,因此安全较低.,我们可以通过加密后保存.
->补讲md5加密算法 :
请大家注意,以后我们的密码都要使用加密存放,在验证密码的时候,对用户输入密码,进行md5加密,然后该到数据库去验证。
md5算法
package com.hsp;
import java.security.*;
import java.security.spec.*;
class MD5_test {
public final static String MD5(String s) {
char hexDigits[] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'a', 'b', 'c', 'd', 'e', 'f' };
try {
byte[] strTemp = s.getBytes();
MessageDigest mdTemp = MessageDigest.getInstance("MD5");
mdTemp.update(strTemp);
byte[] md = mdTemp.digest();
int j = md.length;
char str[] = new char[j * 2];
int k = 0;
for (int i = 0; i < j; i++) {
byte byte0 = md[i];
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
}
return new String(str);
} catch (Exception e) {
return null;
}
}
public static void main(String[] args) {
// MD5_Test aa = new MD5_Test();
System.out.print(MD5_test.MD5("韩顺平"));
}
}
保存上次登录时间
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
PrintWriter out = response.getWriter();
//先获取cookie
// 假设我们 保存上次登录时间的cookie "lasttime" "2011-11-11 12:12:12";
// 这里我们要考虑一个情况: 用户第一次登录 '您是第一次登录..'
Cookie []cookies=request.getCookies();
boolean b=false;//假设没有lasttime cookie
if(cookies!=null){ //保证有cookie,取遍历
for(Cookie cookie: cookies){
//取出名
String name=cookie.getName();
if("lasttime".equals(name)){
//显示
out.println("您上次登录时间是 "+cookie.getValue());
//更新时间
//把当前日期保存cookie
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String nowTime=simpleDateFormat.format(new java.util.Date());
Cookie mycookie=new Cookie("lasttime",nowTime);
mycookie.setMaxAge(7*3600*24);//保存一周
response.addCookie(mycookie);
b=true;
break;
}
}
}
if(!b){
//没有找到
out.println("您是第一次登录..");
//把当前日期保存cookie
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String nowTime=simpleDateFormat.format(new java.util.Date());
Cookie cookie=new Cookie("lasttime",nowTime);
cookie.setMaxAge(7*3600*24);//保存一周
response.addCookie(cookie);
}
}
*上机练习
cookie自动保存用户密码
打开登录页面的时候,自动填写该用户的用户名和密码【这个要求学员课堂练习(最好单开一个项目)】
cookie保存浏览商品
请自己作为作业考虑实现
cookie的细节
① 一个浏览器最多放入 300cookie,一个web站点,最多 20cookie,而且一个cookie大小限制子4k
② cookie生命周期的再说明:
1. cookie默认生命周期是会话级别
2. 通过setMaxAge() 可以设置生命周期
setMaxAge(正数) , 即多少秒后该cookie失效
setMaxAge(0) ,删除该cookie
setMaxAge(负数), 相当于该cookie生命周期是会话级别.
案例 :
//先得到该cookie
Cookie cookies[]=request.getCookies();
for(Cookie cookie: cookies){
if(cookie.getName().equals("id")){
System.out.println("id");
//删除
cookie.setMaxAge(0);
response.addCookie(cookie);//一定带上这句话,否则不能删除
}
}
特别说明: 如果该web应用只有一个cookie ,则删除该cookie后,在浏览器的临时文件夹下没有该cookie文件,如果该web应用有多个cookie,则删除一个cookie后,文件还在,只是该cookie没有
③ cookie存放中文,怎么处理
存放:
String val=java.net.URLEncoder.encode("顺平","utf-8");
Cookie cookie=new Cookie("name",val);
取出:
String val=java.net.URLDecoder.decode(cookie.getValue(), "utf-8");
out.println("name ="+val);
session为什么有?
问题1: 如何实现在不同的页面,可以去查看信息(比如说购物车),同时还要实现不同的用户看到的信息是自己.
session工作原理图
session的生命周期是30分钟
session 小结:
① session是存在服务器的内存中
② 一个用户浏览器,独享一个session域对象
③ session中的属性的默认生命周期是30min ,你可以通过 web.xml来修改
④
3种session生命周期的设置
(1)一个地方是 tomcat/conf/web.xml
对所有的web应用生效
(2)另外一个地方,就是在单个web应用的下去修改 web.xml
如果发生冲突,则以自己的web应用优先级高
(3)session.setMaxInactiveinterval(60)发呆六十秒后session失效
对session和cookie生命周期小结:
⑤ session中可以存放多个属性
⑥ session 可以存放对象
⑦ 如果 session.setAttribute(“name”,val) , 如果名字重复,则会替换该属性.
?如果同一个用户浏览器,向session设置一个属性的时候,如果名字相同了,会出现什么情况?
结论: 会替换该对象值.
session的更深入理解:
为什么服务器能够为不同的浏览器提供不同session?
因为每个浏览器去访问web站点的时候,如果发出的http请求头没有带JSESSIONID头就会自动给你创建一个并返回
www.sourceforge.net [开源之祖]
生成验证码案例
使用(原理是使用到java的绘图技术.)
这里最重要的就是生成验证码的servlet
package com.hsp;
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Random;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class CreateCode extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// 7.禁止浏览器缓存随机图片
response.setDateHeader("Expires", -1);
response.setHeader("Cache-Control", "no-cache");
response.setHeader("Pragma", "no-cache");
// 6.通知客户机以图片方式打开发送过去的数据
response.setHeader("Content-Type", "image/jpeg");
// 1.在内存中创建一副图片
BufferedImage image = new BufferedImage(60, 30,
BufferedImage.TYPE_INT_RGB);
// 2.向图片上写数据
Graphics g = image.getGraphics();
// 设背景色
g.setColor(Color.BLACK);
g.fillRect(0, 0, 60, 30);
// 3.设置写入数据的颜色和字体
g.setColor(Color.RED);
g.setFont(new Font(null, Font.BOLD, 20));
// 4.向图片上写数据
String num = makeNum();
//这句话就是把随机生成的数值,保存到session
request.getSession().setAttribute("checkcode", num); 通过session就可以直接去到随即生成的验证码了
g.drawString(num, 0, 20);
// 5.把写好数据的图片输出给浏览器
ImageIO.write(image, "jpg", response.getOutputStream());
}
//该函数时随机生成7位数字
public String makeNum() {
Random r = new Random();
//9999999 可以生成7位
String num = r.nextInt(9999) + "";
StringBuffer sb = new StringBuffer();
//如果不够4位,前面补零
for (int i = 0; i < 4 - num.length(); i++) {
sb.append("0");
}
num = sb.toString() + num;
return num;
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
}
如何适用
Login.java
练习,请大家把 验证码功能加入到用户管理系统.
过滤器(filter)
①开发过滤器的步骤:
1. 创建 继承HttpServlet 同时实现Filter接口
2. 默认filter不生效,需要配置.
3. 在filter的方法中添加业务逻辑.
package com.hsp.filter;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import com.hsp.domain.User;
public class MyFilter1 extends HttpServlet implements Filter {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
PrintWriter out = response.getWriter();
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doGet(request, response);
}
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.print("myfilter1...");
//获取session
HttpServletRequest httpServletRequest=
(HttpServletRequest)request;
//看看请求的资源是什么
String uri=httpServletRequest.getRequestURI();
if(uri.startsWith("/UsersManager3/imgs")||uri.startsWith("/UsersManager3/Login")){
//直接放行.
chain.doFilter(request, response);
}else{
HttpSession session=httpServletRequest.getSession();
User user=(User) session.getAttribute("loginuser");
if(user!=null){
//该用户合法,放行
chain.doFilter(request, response);
}else{
request.setAttribute("err", "请好好登陆");
httpServletRequest.getRequestDispatcher("/LoginServlet")
.forward(request, response);
}
}
}
public void init(FilterConfig arg0) throws ServletException {
// TODO Auto-generated method stub
}
}
过滤器链
实现方式 :
1. 在创建一个过滤器 (继承HttpServlet 同时还要实现Filter接口)
2. 配置过滤器
配置过滤器的顺序就可以决定调用过滤器的顺序.
控制session的销毁时间
对session的销毁时间的讨论—借助一个案例:
面试题: (应用:关掉IE后,再开IE,上次购买的商品还在。->涉及到session销毁时间)
分析
我们的session 生命周期如果是30min,该session不会随浏览器的关闭,而自动销毁.而会到30min后,才会被服务器销毁.
我们使用代码来实现该功能(session + cookie结合使用)
分析实现的思路:
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
PrintWriter out = response.getWriter();
HttpSession session = request.getSession();
session.setAttribute("name", "张辉胤");
out.println("创一个session并放入姓名属性");
Cookie cookie = new Cookie("JSESSIONID", session.getId());
cookie.setMaxAge(60*30);
response.addCookie(cookie);
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
PrintWriter out = response.getWriter();
HttpSession httpSession = request.getSession();
String name = (String) httpSession.getAttribute("name");
out.println("name = "+name);
}
ie禁用cookie后使用session的方法
如何实现ie禁用cookie后,我们还可以继续使用session.
简易购物车的实例:
思路:
当用户点击购买商品时,我们把该商品保存到session中,该session的结构是
name val
mybookds hashMap对象
而hashmap的结构是
key val
书号 书对象.
Book.java类;
具体的实现,参看myCart项目
总节我们使用到的相关知识
(1) java基础的集合 ArrayList HashMap LinkedHashmap(有序的)
(2) session技术
(3) servlet
(4) 单态
(5) 如何选择不同的集合
list 集合都是有序
map 集合是无序
list和map集合都可以放入null
list可以放入相同的对象,而map也可以放相同对象 , 但是key不能重复.
JSP笔记
1. 为什么需要servletContext
需求1
需求2
解决之道—ServletContext
原理图:
快速入门
ServletContext
1. ServletContext 是在服务器
2. ServletContext 是被所有客户端共享
3. ServletContext 是当web应用启动的时候,自动创建
4. ServletContext 当web应用关闭/tomcat关闭/对web应用reload 会造成servletContext销毁.
对ServletContext的用法小结
获取:
this.getServletContext(); this.getServletConfig().getServletContext();
添加属性:
servletcontext.setAttribute(string,object);
取出属性
servletcontext.getAttribute(“属性名”)
删除
setvletContext.removeAttribute(“属性名”);
ServletContext的应用
(1) 获取WEB应用的初始化参数
如何获取
String val= this.getServletContext().getInitParameter("name");
(2) 使用ServletContext实现跳转
//目前我们跳转到下一个页面
//1 response.sendRedirect("/web应用名/资源名");
//2 request.getRequestDispatcher("/资源名").forward(request, response);
/*
* 区别1. getRequestDispatcher 一个跳转发生在web服务器 sendRedirect发生在浏览器
* 2. 如果request.setAttribute("name","顺平") 希望下一个页面可以使用 属性值,则使用 getRequestDispatcher
* 3. 如果session.setAttribute("name2","顺平3"), 希望下一个页面可以使用 属性值,则两个方法均可使用,但是建议使用 getRequestDispatcher
* 4. 如果我们希望跳转到本web应用外的一个url,应使用sendRedirect
*/
//3.这种方法和2一样
this.getServletContext().getRequestDispatcher("/资源url").forward(request, response);
(3) 读取文件,和获取文件全路径
//首先读取到文件
InputStream inputStream=this.getServletContext().getResourceAsStream("dbinfo.properties");
//创建Properties
Properties pp=new Properties();
pp.load(inputStream);
out.println("name="+pp.getProperty("username"));
*如果文件放在src目录下;则使用类加载器
//如果文件放在src目录下,我们应该使用类加载器来读取
InputStream is=Servlet5.class.getClassLoader().getResourceAsStream("dbinfo.properties")
//获取文件全路径
//如果读取到一个文件的全路径
String path=this.getServletContext().getRealPath("/imgs/Sunset.jpg");
out.println("paht = "+path);
网站计数器的思考
分析 :
代码:
我们建立一个文件recoder.txt文件,用于保存访问量,可以可以保证稳定增长.
实现方法
建立InitServlet ,用于初始化我的Servletcontext,和在关闭tomcat时保存访问量
package com.hsp;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class InitServlet extends HttpServlet {
/**
* Destruction of the servlet.
*/
public void destroy() {
System.out.println("init servlet destory()被调用..");
//把ServletContext值重新保存到文件.
//从record.txt文件中,读取浏览量
//1.首先得到该文件真实路径
String filePath=this.getServletContext().getRealPath("record.text");
//2.打开文件
try {
FileWriter filewriter=new FileWriter(filePath);
//为了读取我们转为BufferedReader
BufferedWriter bufferedWriter=new BufferedWriter(filewriter);
//从ServletContext读取访问量
String nums=(String) this.getServletContext().getAttribute("nums");
//重新写会文件
bufferedWriter.write(nums);
//一定要关闭流
bufferedWriter.close();
filewriter.close();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally{
//关闭...
}
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html");
PrintWriter out = response.getWriter();
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doGet(request, response);
}
public void init() throws ServletException {
// Put your code here
//从record.txt文件中,读取浏览量
//1.首先得到该文件真实路径
String filePath=this.getServletContext().getRealPath("record.text");
//2.打开文件
try {
FileReader fileReader=new FileReader(filePath);
//为了读取我们转为BufferedReader
BufferedReader bufferedReader=new BufferedReader(fileReader);
String nums=bufferedReader.readLine();
//把nums添加到Servletcontext
this.getServletContext().setAttribute("nums", nums);
//一定要关闭流
bufferedReader.close();
fileReader.close();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
当用户登录一次我们取出ServletContext取出,并+1,
//向servletContext中添加属性
//1.先取出
String nums=(String) this.getServletContext().getAttribute("nums");
//如果有,则取出+1
this.getServletContext().setAttribute("nums", (Integer.parseInt(nums)+1)+"");
在Mange页面显示
String nums=this.getServletContext().getAttribute("nums").toString();
out.println("该管理页面被访问了 "+nums+" 次");
问:如果我们的tomcat异常退出,怎么办.
使用线程,定时把ServletContext的值,刷新到recorder.txt 比如10min.
上机练习:
最后说:
针对工具类 SqlHelper.java
1. 我们的链接数据库的变量都是static,这样有一个潜在危险.如果访问量大,可能造成一些用户等待超时.我们可以这样做:
把static变量 改成 非 static
在调用SqlHelper时候,首先创建一个 SqlHelper对象,然后调用相应的方法.
2. 我们的SqlHelper在查询数据的 ResultSet 没有在SqlHelper本类中关闭,不是太好,解决方案入下:
public static ResultSet executeQuery(String sql,String []parameters){
try {
ct=getConnection();
ps=ct.prepareStatement(sql);
if(parameters!=null&&!parameters.equals("")){
for(int i=0;i
}
}
rs=ps.executeQuery();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e.getMessage());
// TODO: handle exception
}finally{
//close(rs, ps, ct);
}
return rs;
}
上面的代码可以修改成这样:
jsp讲解
jsp为什么会出现?
因为在开发web网站时候,发现servlet做界面比较麻烦,于是又有一个新的技术jsp
jsp是什么?
1. jsp运行在服务器
2. jsp(java server page)
3. jsp的基础是servlet(相当于对servlet进行一个包装)
4. jsp是综合技术 jsp=html+css+javascript+java代码+jsp标签(servlet)
5. jsp无需配置.直接使用,如果你修改了jsp文件,不需要重新reload web应用.
6. jsp如何方法 http://ip:8088/web应用名/jsp路径
7. jsp是一种动态网页技术.
快如入门案例
讲解jsp的运行原理
Web服务器是如何调用并执行一个jsp页面的?
看上面的原理图
Jsp页面中的html排版标签是如何被发送到客户端的?
out.write(" \r\n");
\r\n");
out.write(" \r\n");apple melon orange
out.write(" \r\n");apple melon orange
out.write(" \r\n");apple melon orange
out.write(" \r\n");apple melon orange
out.write("
Jsp页面中的java代码服务器是如何执行的?
比如jsp是:
<%
int i=90;
int j=i+90;
%>
测试.
<%
out.println("j="+j);
%>
当被翻译成jsp后:java文件
public jspService(){
int i=90;
int j=i+90;
}
1. 就是有多个<% %> 其实相当于是一个大的 <% %>
2. 在<% %> 中定义的变量,会成为service函数的局部变量.
Web服务器在调用jsp时,会给jsp提供一些什么java对象?
供提供了九个: 我们使用了 out 对象 --->Servlet 的 PrintWriter
说 jsp=html+css+js+java代码+jsp语法(标签 )
jsp的语法
① 指令元素
概念: 用于从jsp发送一个信息到容器,比如设置全局变量,文字编码,引入包
1. page指令
<%@ page contentType="text/html;charset=gb2312"%>
常用的属性:
contentType 和 pageEncoding的区别
contentType=“text/html;charset=utf-8” 指定网页以什么方式显示页面
pageEncoding=“utf-8” 指定Servlet引擎以什么方法翻译jsp->servlet 并 指定网页以什么方式显示页面
2. include指令
用法:
<%@ include file=”文件路径” %>
3. taglib指令
② 脚本元素
java片段:
<% java 代码 %>
表达式:
<%=表达式 %>
<%
int i=90;
%>
显示
<%
out.println("i="+i);
%>
<%=i*78-23 %>
定义变量
<%! int i=90; %>
定义函数
<%! public int getResult(int a,int b){return a+b;}%>
函数不能在
<% %> 定义.
面试题:
③ 动作元素
在开发jsp的过程中,我们通常把jsp放入WEB-INF目录,目的是为了防止用户直接访问这些jsp文件.
在WebRoot下我们有一个入口页面,它的主要转发
动态引入:
<%@ include file=””%> 静态引入
相同点: 把一个文件引入到另外一个文件
区别:静态引入 把两个jsp翻译成一个Servlet,所以被引入的文件不要包含..
动态引入 把两个jsp分别翻译,所以被引入的jsp包含有也可以.
修改jsp的模板:
mvc (m(model模型) v (view视图) c(controller控制器)
mvc 它要求程序员做开发把 数据的输入(使用jsp 视图),数据的处理(使用Servlet 即 Controller 调用model完成.),数据的显示(使用jsp),分开.
iso9001
作业:
改写项目成mvc
JSP的9个内置对象
对象名 类型 作用域
request:请求对象 javax.servlet.ServletRequest的子类 Request
response:响应对象 javax.servlet.ServletResponse的子类 Page
pageContext:页面上下文对象 javax.servlet.jsp.PageContext Page
session:会话对象 javax.servlet.http.HttpSession Session
application:应用程序对象 javax.servlet.ServletContext Application
out:输出对象 javax.servlet.jsp.JspWriter Page
config:配置对象 javax.servlet.ServletConfig Page
page:页面对象 java.lang.Object Page
exception:异常对象 java.lang.Throwable Page
getParameter
getParameterNames
getParameterValues