Servlet
- Servlet是JavaWeb的三大组件之一(Servlet、Filter、Listener),它属于动态资源。Servlet的作用是处理请求,服务器会把接收到的请求交给Servlet来处理,在Servlet中通常需要:接收请求数据 处理请求 完成响应。
- 实现servlet的方式
实现javax.servlet.Servlet接口;
继承javax.servlet.GenericServlet类;
继承javax.servlet.http.HttpServlet类; - servlet接口里的方法
public void init(ServletConfig config) throws ServletException
public ServletConfig getServletConfig();
public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException;
public String getServletInfo();
public void destroy();
}
生命周期方法
- void init(ServletConfig):创建之后(1次);
- void service(ServletRequest request, ServletResponse response):每次处理请求时都会被调用;(多次)
- void destroy():消亡之前(1次);
特性:
- 单例,一个类只有一个对象;当然可能存在多个Servlet类!
- 线程是不并行的,所以它的效率是高的!
Servlet类由我们来写,但对象由服务器来创建,并且由服务器来调用相应的方法。
非生命周期方法
- ServletConfig是Servlet中的init()方法的参数类型,服务器会在调用init()方法时传递ServletConfig对象给init()方法。ServletConfig对象封装了Servlet在web.xml中的配置信息,它对应
<servlet>
元素。 - ServletConfig类的功能有:
String getServletName():获取Servlet配置名,即的值;
ServletContext getServletContext():获取ServletContext对象
String getInitParameter(String name):获取初始化参数
Enumeration getInitParameterNames():获取所有初始化参数的名称 <servlet>
的子标签有:
<servlet-name>
<servlet-class>
<init-param>
<load-on-startup>
HttpServlet理解
- HttpServlet是GenericServlet的子类,实现了GenericServlet中的抽象方法
- service(ServletRequest, ServletResponse)会调用本类的service(HttpServletRequest,HttpServletResponse)
- service(HttpServletRequest,HttpServletResponse)方法会根请求方式来调用相对的处理方法,例如请求方式为GET,那么该方法会调用doGet()方法,请求方式是POST,那么会调用doPost()方法。所以我们应该去重写HttpServlet的doGet()或doPost()方法。
ServletContext理解
- 一个Web应用只有一个ServletContext对象,它会在服务器启动时创建,会在服务器关闭时被销毁,它的生命与服务器相同。
- 通常在整个Web应用中共享数据时可以使用ServletContext对象
- 还可以使用ServletContext来获取Web资源的真实路径,servletContext.getRealPath("/WEB-INF/a.jpg");
- ServletContext可以用来资源的MIME类型,例如:servletContext.getMimeType(“a.jpg”),它会返回image/jpeg
Servlet与线程安全
因为一个类型的Servlet只有一个实例对象,那么就有可能会现时出一个Servlet同时处理多个请求,所以Servlet不是线程安全的,但是作效率很高!所以我们不应该在Servlet中便宜创建成员变量,因为可能会存在一个线程对这个成员变量进行写操作,另一个线程对这个成员变量进行读操作。
如果要创建变量,必须遵循以下规则:
不要在Servlet中创建成员!创建局部变量即可!
可以创建无状态成员!
可以创建有状态的成员,但状态必须为只读的!
- javaWeb四大域
PageContext;
ServletRequest;
HttpSession;
ServletContext
获取资源相关方法
- 获取a.txt的真实路径:String realPath = servletContext.getRealPath(“/a.txt”),
realPath的值为a.txt文件的绝对路径:F:\tomcat6\webapps\hello\a.txt; - 获取资源流
不只可以获取资源的路径,还可以通过ServletContext获取资源流,即把资源以输入流的方式获取:
获取a.txt资源流:InputStream in = servletContext.getResourceAsStream(“/a.txt”);
获取b.txt资源流:InputStream in = servletContext.getResourceAsStream(“/WEB-INF/b.txt”); - 获取指定路径下的所有路径文件
Set set = context.getResourcePaths("/WEB-INF");(开头必须以 / 开头) - 可以通过Class类的对象来获取类路径下的资源,对应JavaWeb应用的类路径就是classes目录下的资源
例如:
InputStream in = projectname.MyServlet.class.getResourceAsStream(“a.jpg”);
获取的是:/WEB-INF/classes/servlet/a.jpg,即与MyServlet.class同目录下的资源
例如:
InputStream in = projectnaem.servlet.MyServlet.class.getResourceAsStream("/a.jpg");
获取的是:/WEB-INF/classes/a.jpg,即类路径的根目录下的资源,类路径的根目录就是/classes目录 - 如果我们的 servlet-mapping 如下配置:
<servlet-mapping>
<servlet-name>jetbrick-template</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
那么访问: /context/index.jsp
request.getServletPath() == ""
request.getPathInfo() == "/templates/index.jsp"
- 如果我们的 servlet-mapping 如下配置:
<servlet-mapping>
<servlet-name>jetbrick-template</servlet-name>
<url-pattern>/template/*</url-pattern>
</servlet-mapping>
那么访问: /context/templates/index.jsp
request.getServletPath() == "/templates"
request.getPathInfo() == "/index.jsp"
request和response
- 当服务器接收到请求后,服务器会创建request和response对象,把请求数据封装到request对象中;
- 然后调用Servlet的service()方法时把这两个对象传递给service()方法;
- 在service()方法中可以通过request对象获取请求数据,可以使用response对象向客户端完成响应;
- 每次请求服务器都会创建新的request和response对象,即每个请求有自己独自的request和response对象。
response
- response是Servlet.service方法的一个参数,类型为javax.servlet.http.HttpServletResponse。在客户端发出每个请求时,服务器都会创建一个response对象,并传入给Servlet.service()方法。response对象是用来对客户端进行响应的,这说明在service()方法中使用response对象可以完成对客户端的响应工作。
- response对象的功能分为以下四种:
设置响应头信息;
发送状态码;
设置响应正文;
重定向;
response响应正文:
repsonse一共提供了两个响应流对象:不能同时使用这两个流,否则会抛异常
- PrintWriter out = response.getWriter():获取字符流;
- ServletOutputStream out = response.getOutputStream():获取字节流;
设置响应头信息
响应头:Content-Type、Refresh、Location等等
- 可以使用response对象的setHeader()方法来设置响应头!使用该方法设置的响应头最终会发送给客户端浏览器!
response.setHeader(“Refresh”,"5; URL=http://www.baidu.com):5秒后自动跳转到百度主页。 - response.setContentType(“text/html;charset=utf-8”):等同与调用response.setHeader(“content-type”, “text/html;charset=utf-8”);
- response.setCharacterEncoding(“utf-8”):设置字符响应流的字符编码为utf-8
setHeader(String name, String value):适用于单值的响应头
addHeader(String name, String value):适用于多值的响应头
setIntHeader(String name, int value):适用于单值的int类型的响应头
response.setIntHeader(“Content-Length”, 888);
addIntHeader(String name, int value):适用于多值的int类型的响应头
setDateHeader(String name, long value):适用于单值的毫秒类型的响应头
response.setDateHeader(“expires”, 1000 * 60 * 60 * 24); - addDateHeader(String name, long value):适用于多值的毫秒类型的响应头;
设置状态码
- response.setStatus(200):设置状态码;
- response.sendError(404, “您要查找的资源不存在”):当发送错误状态码时,Tomcat会跳转到固定的错误页面去,但可以显示错误信息。
- response.setStatus(200):设置状态码为200
response.sendError(404, “您要查找的资源不存在”):设置状态码为404
repsonse.sendError(500, “服务器出氏了”):设置状态码为500
在调用sendError()方法时,不只是设置了状态码,而且还会给浏览器一个显示错误的页面。
重定向
- 重定向:一个快捷方法,完成重定向!
sendRedirect(String location)方法:
例: response.sendRedirect(“http://www.baidu.com”); - response.sendStatus(302);
repsonse.setHeader(“Location”, “http://www.baidu.com”); - 总结:
重定向是两次请求;
重定向的URL可以是其他应用,不局限于当前应用;
重定向的响应头为302,并且必须要有Location响应头;
重定向就不要再使用response.getWriter()或response.getOutputStream()输出数据,不然可能会出现异常;
response字符编码
- Tomcat响应数据默认使用ISO-8859-1
- 通常浏览器默认使用GBK编码
- response.setCharacterEncoding(“utf-8”);
response.setContentType(“text/html;charset=utf-8”);
response.getWriter().print(“大家好”);
setContentType()方法有两个作用: - 设置字符流编码。等同与调用了response.setCharacterEncoding(“utf-8”);
- 设置Content-type响应头,即通知浏览器响应数据的编码为utf-8。
因为设置字符流的编码为utf-8,所以响应给客户端数据为utf-8编码
因为设置了Content-type头为utf-8,所以浏览器会使用utf-8来解析响应数据 - response.setHeader(“Content-type”, “text/html;charset=utf-8”)
等同于
repsonse.setContentType(“text/html;charset=utf-8”)
request
- request是Servlet.service()方法的一个参数,类型为javax.servlet.http.HttpServletRequest。在客户端发出每个请求时,服务器都会创建一个request对象,并把请求数据封装到request中,然后在调用Servlet.service()方法时传递给service()方法,这说明在service()方法中可以通过request对象来获取请求数据。
- request的功能可以分为以下几种:
封装了请求头数据;
封装了请求正文数据,如果是GET请求,那么就没有正文;
request是一个域对象,可以把它当成Map来添加获取数据;
request提供了请求转发和请求包含功能。 - request是域对象
在JavaWeb中一共四个域对象,其中ServletContext就是域对象,它在整个应用中只创建一个ServletContext对象。request其中一个,request可以在一个请求中共享数据。
一个请求会创建一个request对象,如果在一个请求中经历了多个Servlet,那么多个Servlet就可以使用request来共享数据 - 获取常用信息
获取客户端IP,案例:封IP。request.getRemoteAddr()
请求方式,request.getMethod(),可能是POST也可能是GET - 获取HTTP请求头
String getHeader(String name),适用于单值头
int getIntHeader(String name),适用于单值int类型的请求头
long getDateHeader(String name),适用于单值毫秒类型的请求头
Enumeration getHeaders(String name),适用于多值请求头 - 获取请求URL
String getScheme():获取协议,http
String getServerName():获取服务器名,localhost
String getServerPort():获取服务器端口,8080
String getContextPath():获取项目名,/day10_2
String getServletPath():获取Servlet路径,/AServlet
String getQueryString():获取参数部分,即问号后面的部分。username=xxx&password=yyy
String getRequestURI():获取请求URI,等于项目名+Servlet路径。/day10_2/AServlet
String getRequestURL():获取请求URL,等于不包含参数的整个请求路径。http://localhost:8080/day10_2/AServlet - 获取请求参数:请求参数是由客户端发送给服务器的!有可能是在请求体中(POST),也可能是在URL之后(GET)
String[] getParameterValues(String name):获取指定名称的请求参数值,适用于多值请求参数
Enumeration<String>
getParameterNames():获取所有请求参数名称
*Map<String,String[]>
getParameterMap():获取所有请求参数,其中key为参数名,value为参数值。 - 请求转发和请求包含
RequestDispatcher rd = request.getRequestDispatcher("/MyServlet"); 使用request获取RequestDispatcher对象,方法的参数是被转发或包含的Servlet的Servlet路径
请求转发:rd.forward(request,response);
请求包含:rd.include(request,response);
请求转发:由下一个Servlet完成响应体!当前Servlet可以设置响应头!(留头不留体)
请求包含:由两个Servlet共同未完成响应体!(都留)
无论是请求转发还是请求包含,都在一个请求范围内!使用同一个request和response! - request域
Servlet中三大域对象:request、session、application,都有如下三个方法:
void setAttribute(String name, Object value)
Object getAttribute(String name)
void removeAttribute(String name);
请求转发和重定向的异同
- 请求转发是通过RequestDispatcher对象的forward()方法完成的
- 重定向是通过HttpServletResponse对象的sendRediect()方法完成的
- 请求转发是一个请求一次响应,而重定向是两次请求两次响应
- 请求转发是在一个请求中跨越多个动态资源(jsp/servlet),所以多个动态资源之间可以共享request数据
- 重定向是两次请求,第一次请求服务器响应给客户端的是302,以及Location响应头,通知客户端再次去请求新的资源,所以客户端又发出第二次请求。所以重定向中被请求的多个动态资源之间不能共享request数据。
- 请求转发地址栏不变化,而重定向会显示后一个请求的地址
- 请求转发只能转发到本项目其他Servlet,而重定向不只能重定向到本项目的其他Servlet,还能定向到其他项目
- 请求转发是服务器端行为,只需给出转发的Servlet路径,而重定向需要给出requestURI,即包含项目名!
- 请求转发和重定向效率是转发高!因为是一个请求!
- 重定向的第二个请求一定是GET;
- 最为常见的客户端传递参数方式有两种:
浏览器地址栏直接输入:一定是GET请求;
超链接:一定是GET请求;
表单:可以是GET,也可以是POST,这取决与的method属性值;
乱码
首先,使用让字符串通过iso-8859-1返回到字节数据,即还原字节数据,然后在重新使用正确的utf-8来解码。
String username = new String(request.getParameter(“s”).getBytes(“iso-8859-1”), “utf-8”);
路径
- web.xml中路径,(叫它Servlet路径!) 以“/”开头
- 转发和包含路径:
以“/”开头:相对当前项目路径,
不以“/”开头:相对当前Servlet路径。 - 重定向路径(客户端路径)
以“/”开头:相对当前主机 - 页面中超链接和表单路径
与重定向相同,都是客户端路径!需要添加项目名 - ServletContext获取资源路径
相对当前项目目录,即当然index.jsp所在目录 - ClassLoader获取资源路径
相对classes目录 - Class获取资源路径
以“/”开头相对classes目录
不以“/”开头相对当前.class文件所在目录。