Servlet的运行流程
1) Web服务器首先检查是否已经装载并创建了该Servlet的实例对象。如果是,则直接执行第4)步,否则,执行第2)步;
2) 装载并创建该Servlet的一个实例对象;
3) 调用Servlet实例对象的init()方法;
4) 创建一个用于封装HTTP请求消息的HttpServletRequest对象和一个代表HTTP响应消息的HttpServletResponse对象,然后调用Servlet的service()方法并将请求和响应对象作为参数传递进去;
5) 在service()方法中使用request获取请求数据,使用response完成相应;
6) Web应用程序被停止或重新启动之前,Servlet引擎将卸载Servlet,并在卸载之前调用Servlet的destroy()方法。
response响应对象在调用service方法之前就已经创建,并不是到真正相应的时候才创建,每次请求服务器都会创建新的request和response对象,即每个请求有自己独自的request和response对象。response
response是Servlet.service方法的一个参数,类型为javax.servlet.http.HttpServletResponse。在客户端发出每个请求时,服务器都会创建一个response对象,并传入给Servlet.service()方法。response对象是用来对客户端进行响应的,这说明在service()方法中使用response对象可以完成对客户端的响应工作。
response对象的功能有以下四种:
1. 设置响应正文;
2. 设置响应头信息;
3. 发送状态码;
4. 重定向。
response是响应对象,向客户端输出响应正文(响应体)可以使用response的响应流,repsonse一共提供了两个响应流对象:
1) PrintWriter out =response.getWriter():获取字符流
1) ServletOutputStream out =response.getOutputStream():获取字节流
可以使用response对象的setHeader()方法来设置响应头!使用该方法设置的响应头最终会发送给客户端浏览器。
response.setHeader(“content-type”,“text/html;charset=utf-8”):设置content-type响应头,该头的作用是告诉浏览器响应内容为html类型,编码为utf-8。而且同时会设置response的字符流编码为utf-8,即response.setCharaceterEncoding(“utf-8”),Ø response.setHeader("Refresh","5; URL=http://www.baidu.com"):5秒后自动跳转到百度。
Ø 如何实现重定向
1) 设置响应码为302,302表示重定向;
2) 重定向是通知浏览器再第二个请求,所以浏览器需要知道第二个请求的URL,完成重定向的第二步是设置Location头,指定第二个请求的URL地址;
public class DServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//实现重定向
//1.设置响应码为302
response.setStatus(302);
//2.设置第二个请求的URL地址
response.setHeader("Location","/Requestresponse/index.jsp");
}
}
Ø 实现重定向的便捷方式(常用方式)
response.sendRedirect()方法会设置响应头为302,以设置Location响应头public class DServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//实现重定向的便捷方式
response.sendRedirect("/Requestresponse/index.jsp");
}
}
Ø 重定向小结
1) 重定向是两次请求;
2) 重定向的URL可以是其他应用,不局限于当前应用;
3) 重定向的响应头为302,并且必须要有Location响应头;
4) 重定向就不要再使用response.getWriter()或response.getOutputStream()输出数据,不然可能会出现异常。
request
request是Servlet.service()方法的一个参数,类型为javax.servlet.http.HttpServletRequest。在客户端发出每个请求时,服务器都会创建一个request对象,并把请求数据封装到request中,然后再调用Servlet.service()方法时传递给service()方法,这说明在service()方法中可以通过request对象来获取请求数据。request包含客户端的所有请求信息。
request的功能如下:
Ø 封装了请求头数据;
Ø 封装了请求正文数据,如果是GET请求,那么就没有正文;
Ø request是一个域对象,可以把它当成Map来添加获取数据;
request提供了请求转发和请求包含功能。
request的域方法:
voidsetAttribute(String name, Object value):用来存储一个对象,也可以称之为存储一个域属性Enumeration getAttributeNames():获取所有域属性的名称。
Ø String getHeader(String name):获取指定名称的请求头;
Ø Enumeration getHeaderNames():获取所有请求头名称;
Ø int getIntHeader(String name):获取值为int类型的请求头。
int getContentLength():获取请求体的字节数,GET请求没有请求体,没有请求体返回-1;
String getContentType():获取请求类型,如果请求是GET,那么这个方法返回null;如果是POST请求,那么默认为application/x-www-form-urlencoded,表示请求体内容使用了URL编码;
String getMethod():返回请求方法,例如:GET
Locale getLocale():返回当前客户端浏览器的Locale。java.util.Locale表示国家和言语,这个东西在国际化中很有用;
String getCharacterEncoding():获取请求编码,如果没有setCharacterEncoding(),那么返回null,表示使用ISO-8859-1编码;
void setCharacterEncoding(String code):设置请求编码,只对请求体有效!注意,对于GET而言,没有请求体!!!所以此方法只能对POST请求中的参数有效!
String getContextPath():返回上下文路径,例如:/Requestresponse
String getQueryString():返回请求URL中的参数,例如:name=zhangSan
String getRequestURI():返回请求URI路径,例如:/Requestresponse /servlet/FServlet
StringBuffer getRequestURL():返回请求URL路径,例如:http://localhost:8080/ Requestresponse /FServlet,即返回除了参数以外的路径信息;
String getServletPath():返回Servlet路径,例如:/Requestresponse /servlet/FServlet
String getRemoteAddr():返回当前客户端的IP地址;
String getRemoteHost():返回当前客户端的主机名,但这个方法的实现还是获取IP地址;
String getScheme():返回请求协议,例如:http;
String getServerName():返回主机名,例如:localhost
int getServerPort():返回服务器端口号,例如:8080
public class FServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("request.getContentLength(): " + request.getContentLength());
System.out.println("request.getContentType(): " + request.getContentType());
System.out.println("request.getContextPath(): " + request.getContextPath());
System.out.println("request.getMethod(): " + request.getMethod());
System.out.println("request.getLocale(): " + request.getLocale());
System.out.println("request.getQueryString(): " + request.getQueryString());
System.out.println("request.getRequestURI(): " + request.getRequestURI());
System.out.println("request.getRequestURL(): " + request.getRequestURL());
System.out.println("request.getServletPath(): " + request.getServletPath());
System.out.println("request.getRemoteAddr(): " + request.getRemoteAddr());
System.out.println("request.getRemoteHost(): " + request.getRemoteHost());
System.out.println("request.getRemotePort(): " + request.getRemotePort());
System.out.println("request.getScheme(): " + request.getScheme());
System.out.println("request.getServerName(): " + request.getServerName());
System.out.println("request.getServerPort(): " + request.getServerPort());
}
}
表单:可以是GET,也可以是POST,这取决与
Map map = request.getParameterMap();
Set keys = map.keySet();
for (String key : keys) {
System.out.println("key:" + key + "values" + Arrays.toString(map.get(key)));
}
Ø 请求转发
在HServlet中,把请求转发到IServlet:
public class HServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("HServlet");
//获取调度器,为调度器绑定要转发到的地址
RequestDispatcher rd = request.getRequestDispatcher("/servlet/IServlet");
//转发
rd.forward(request, response);
}
}
public class IServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("IServlet");
}
}
Ø 请求包含
在HServlet中,把请求包含到IServlet:
public class HServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("HServlet");
//获取调度器,为调度器绑定要转发到的地址
RequestDispatcher rd = request.getRequestDispatcher("/servlet/IServlet");
//包含
rd.include(request, response);
}
}
public class IServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("IServlet");
}
}
请求转发与请求包含比较
1) 如果在AServlet中请求转发到BServlet,那么在AServlet中就不允许再输出响应体,即不能再使用response.getWriter()和response.getOutputStream()向客户端输出,这一工作应该由BServlet来完成;如果是使用请求包含,那么没有这个限制;
2) 请求转发虽然不能输出响应体,但还是可以设置响应头的,例如:response.setContentType(”text/html;charset=utf-8”0);
3) 请求包含大多是应用在JSP页面中,完成多页面的合并;
4) 请求转发大多是应用在Servlet中,转发目标大多是JSP页面;
请求转发和重定向比较(重点)
1) 请求转发是一个请求,而重定向是两个请求;
2) 请求转发后浏览器地址栏不会有变化,而重定向会有变化,因为重定向是两个请求;
3) 请求转发的目标只能是本应用中的资源,重定向的目标可以是其他应用;
4) 请求转发对AServlet和BServlet的请求方法是相同的,即要么都是GET,要么都是POST,因为请求转发是一个请求;
5) 重定向的第二个请求一定是GET;
Ø 超链接
Ø 表单
Ø 转发
Ø 包含
Ø 重定向
Ø
服务器端路径必须是相对路径,不能是绝对路径。但相对路径有两种形式:
Ø 以“/”开头;
Ø 不以“/”开头。
其中请求转发、请求包含都是服务器端路径,服务器端路径与客户端路径的区别是:
Ø 客户端路径以“/”开头:相对当前主机;
Ø 服务器端路径以“/”开头:相对当前应用。
常见字符编码:iso-8859-1(拉丁字母、不支持中文)、big5、gb2312、gb18030、gbk (系统默认编码,中国的国标码)、utf-8(万国码,支持全世界的编码,所以我们使用这个)。
计算机存储的都是二进制,字符必须以数字(二进制)的形式在计算机中储存,所有的字符必须有对应的编码才能存储。同一个字符在不同编码体系中多数时候对应的数字是不同的。
字符在进行编码的时候采用一种方式,在取出进行解码的时候采用另外一种方式进行转换,编码和解码采用不同的编码体系,这就造成了乱码。