Servlet简介
1、JavaServlet是和平台无关的服务器端组件,它运行在servlet容器中。servlet容器负责servlet和客户的通信以及调用servlet的方法,servlet和客户的通信采用“请求响应”的模式。
2、servlet可以完成以下功能
---创建并返回基于客户请求的动态HTML页面。
---创建可嵌入到现有HTML页面中的部分HTML页面(HTML片段)。
---与其他服务器资源(数据库或者Java应用程序)进行通信。
Servlet容器响应客户请求的过程
<!-- 配置和映射servlet -->
<servlet>
<!-- servlet 注册的名字 -->
<servlet-name>helloServlet</servlet-name>
<!-- servlet 的全类名 -->
<servlet-class>com.jsu.servlet.FirstServlet</servlet-class>
</servlet>
<servlet-mapping>
<!-- 需要和某一个servlet 节点的servlet-name 子节点的文本节点一致 -->
<servlet-name>helloServlet</servlet-name>
<!-- 映射具体的访问路径:/代表当前WEB 应用的根目录 -->
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<servlet>
<!-- servlet 注册的名字 -->
<servlet-name>helloServlet</servlet-name>
<!-- servlet 的全类名 -->
<servlet-class>com.jsu.servlet.FirstServlet</servlet-class>
</servlet>
<servlet-mapping>
<!-- 需要和某一个servlet 节点的servlet-name 子节点的文本节点一致 -->
<servlet-name>helloServlet</servlet-name>
<!-- 映射具体的访问路径:/代表当前WEB 应用的根目录 -->
<url-pattern>/hello</url-pattern>
</servlet-mapping>
3.Servlet容器:运行Servlet、JSP、Filter等的软件环境
1)可以来创建servlet,并调用Servlet的相关生命周期。
4.Servlet生命周期的方法:以下方法都是Servlet容器负责的调用,不用手动调用
1)构造器:只被调用一次,只有第一次请求Servlet时,创建Servlet实例,调用构造器。这说明Servlet是单实例的!
2)init方法:只被调用一次,在创建好实例后立即被调用,用于初始化当前Servlet。
3)service:被多次调用,每次请求都会调用service方法,实际用于请求响应。
4)destory:只被调用一次,在当前Servlet所在的WEB应用被卸载前调用,用于释放当前Servlet所占用的资源。
5. load-on-startup 参数:
1)配置在Servlet节点
2) load-on-starup:可以指定Servlet被创建的时机。若为负数,则在第一次请求是被创建,若为0或正数则在当前WEB应用被Servlet容器加载时创建实例,且数越小越早被创建。
Servlet容器响应客户请求的过程
Servlet的注册和运行
Servlet映射的细节
7.ServletConfig:封装了Servlet的配置信息,并且可以获取ServletContext对象
1)
<!-- 配置servlet初始化参数 且该节点必须在load-on-startup前面-->
<init-param>
<!-- 参数名 -->
<param-name>user</param-name>
<!-- 参数值 -->
<param-value>root</param-value>
</init-param>
<init-param>
<param-name>password</param-name>
<param-value>123</param-value>
</init-param>
<init-param>
<!-- 参数名 -->
<param-name>user</param-name>
<!-- 参数值 -->
<param-value>root</param-value>
</init-param>
<init-param>
<param-name>password</param-name>
<param-value>123</param-value>
</init-param>
2)获取初始化参数
》getinitParameter(String name):获取指定参数名的初始化参数
》getinitParameterNames():获取参数名组成的Enumeration对象
3)获取Servlet的配置名称
String name = servletConfig.getInitParameter("user");
System.out.println(name);
String password = servletConfig.getInitParameter("password");
System.out.println(password);
Enumeration<String> names = servletConfig.getInitParameterNames();
while(names.hasMoreElements()){
name = names.nextElement();
System.out.println(name);
}
System.out.println(name);
String password = servletConfig.getInitParameter("password");
System.out.println(password);
Enumeration<String> names = servletConfig.getInitParameterNames();
while(names.hasMoreElements()){
name = names.nextElement();
System.out.println(name);
}
4)获取ServletContext
ServletContext context = servletConfig.getServletContext();
8.ServletContext
1)可以由ServletConfig获取
2)该对象代表当前WEB应用:可以认为ServletContext为当前WEB应用的大管家,可以从中获取到当前WEB应用的各个方面的信息
①获取当前WEB应用的初始化参数
方法:getInitParameter、getInitParameterNames
代码:
ServletContext context = servletConfig.getServletContext();
String name = context.getInitParameter("user");
Enumeration<String> names =context.getInitParameterNames();
while(names.hasMoreElements()){
name = names.nextElement();
System.out.println(name);
}
②获取当前WEB应用某一文件在服务器上的绝对路径而不是部署前的路径
getRealPath(String path);
③获取当前WEB应用的名称
getContextPath();
④获取当前WEB应用的某一文件对应的输入流
getResourceAsStream(String path):path为当前WEB应用的根目录。
9.ServletRequest:封装了请求信息,可以从中得到任何的请求信息
①获取请求参数
>> String getParameter(String name);根据请求参数的名字,返回参数值
>> String[] getParameterValues(String name);根据请求参数的名字,返回字符串数组
>> Map getParameterMap():返回请求参数的键值对:key:参数名,value:参数值 String数组类型
Map<String, String[]> map = request.getParameterMap();
for(Map.Entry<String, String[]> entry:map.entrySet()){
System.out.println("**" + entry.getKey() + ":" + Arrays.asList(entry.getValue()));
}
for(Map.Entry<String, String[]> entry:map.entrySet()){
System.out.println("**" + entry.getKey() + ":" + Arrays.asList(entry.getValue()));
}
>> Enumeration getParameterNames():返回参数名对应的Enumeration对象
②获取请求的URI
③获取请求方式
10.ServletResponse:封装了响应信息,如果想给用户什么响应,具体可以使用该接口方法实现
① getWriter():返回PrintWriter对象,调用该对象的print()方法,将print()中的参数打印到浏览器中。
② 设置响应的类型:response.setContentType("text/html");
③ sendRedir()响应重定向
11. GenericServlet:
1). 是一个 Serlvet. 是 Servlet 接口和 ServletConfig 接口的实现类. 但是一个抽象类. 其中的 service 方法为抽象方法
2). 如果新建的 Servlet 程序直接继承 GenericSerlvet 会使开发更简洁.
3). 具体实现:
①. 在 GenericServlet 中声明了一个 SerlvetConfig 类型的成员变量, 在 init(ServletConfig) 方法中对其进行了初始化
②. 利用 servletConfig 成员变量的方法实现了 ServletConfig 接口的方法
③. 还定义了一个 init() 方法, 在 init(SerlvetConfig) 方法中对其进行调用, 子类可以直接覆盖 init() 在其中实现对 Servlet 的初始化.
④. 不建议直接覆盖 init(ServletConfig), 因为如果忘记编写 super.init(config); 而还是用了 SerlvetConfig 接口的方法,
则会出现空指针异常.
⑤. 新建的 init(){} 并非 Serlvet 的生命周期方法. 而 init(ServletConfig) 是生命周期相关的方法.
public abstract class GenericServlet implements Servlet, ServletConfig {
/** 以下方法为 Servlet 接口的方法 **/
@Override
public void destroy() {}
@Override
public ServletConfig getServletConfig() {
return servletConfig;
}
@Override
public String getServletInfo() {
return null;
}
private ServletConfig servletConfig;
@Override
public void init(ServletConfig arg0) throws ServletException {
this.servletConfig = arg0;
init();
}
public void init() throws ServletException{}
@Override
public abstract void service(ServletRequest arg0, ServletResponse arg1)
throws ServletException, IOException;
/** 以下方法为 ServletConfig 接口的方法 **/
@Override
public String getInitParameter(String arg0) {
return servletConfig.getInitParameter(arg0);
}
@Override
public Enumeration getInitParameterNames() {
return servletConfig.getInitParameterNames();
}
@Override
public ServletContext getServletContext() {
return servletConfig.getServletContext();
}
@Override
public String getServletName() {
return servletConfig.getServletName();
}
}
1). 是一个 Serlvet. 是 Servlet 接口和 ServletConfig 接口的实现类. 但是一个抽象类. 其中的 service 方法为抽象方法
2). 如果新建的 Servlet 程序直接继承 GenericSerlvet 会使开发更简洁.
3). 具体实现:
①. 在 GenericServlet 中声明了一个 SerlvetConfig 类型的成员变量, 在 init(ServletConfig) 方法中对其进行了初始化
②. 利用 servletConfig 成员变量的方法实现了 ServletConfig 接口的方法
③. 还定义了一个 init() 方法, 在 init(SerlvetConfig) 方法中对其进行调用, 子类可以直接覆盖 init() 在其中实现对 Servlet 的初始化.
④. 不建议直接覆盖 init(ServletConfig), 因为如果忘记编写 super.init(config); 而还是用了 SerlvetConfig 接口的方法,
则会出现空指针异常.
⑤. 新建的 init(){} 并非 Serlvet 的生命周期方法. 而 init(ServletConfig) 是生命周期相关的方法.
public abstract class GenericServlet implements Servlet, ServletConfig {
/** 以下方法为 Servlet 接口的方法 **/
@Override
public void destroy() {}
@Override
public ServletConfig getServletConfig() {
return servletConfig;
}
@Override
public String getServletInfo() {
return null;
}
private ServletConfig servletConfig;
@Override
public void init(ServletConfig arg0) throws ServletException {
this.servletConfig = arg0;
init();
}
public void init() throws ServletException{}
@Override
public abstract void service(ServletRequest arg0, ServletResponse arg1)
throws ServletException, IOException;
/** 以下方法为 ServletConfig 接口的方法 **/
@Override
public String getInitParameter(String arg0) {
return servletConfig.getInitParameter(arg0);
}
@Override
public Enumeration getInitParameterNames() {
return servletConfig.getInitParameterNames();
}
@Override
public ServletContext getServletContext() {
return servletConfig.getServletContext();
}
@Override
public String getServletName() {
return servletConfig.getServletName();
}
}
12. HttpServlet:
1). 是一个 Servlet, 继承自 GenericServlet. 针对于 HTTP 协议所定制.
2). 在 service() 方法中直接把 ServletReuqest 和 ServletResponse 转为 HttpServletRequest 和 HttpServletResponse.
并调用了重载的 service(HttpServletRequest, HttpServletResponse)
在 service(HttpServletRequest, HttpServletResponse) 获取了请求方式: request.getMethod(). 根据请求方式有创建了
doXxx() 方法(xxx 为具体的请求方式, 比如 doGet, doPost)
@Override
public void service(ServletRequest req, ServletResponse res)
throws ServletException, IOException {
HttpServletRequest request;
HttpServletResponse response;
try {
request = (HttpServletRequest) req;
response = (HttpServletResponse) res;
} catch (ClassCastException e) {
throw new ServletException("non-HTTP request or response");
}
service(request, response);
}
public void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
//1. 获取请求方式.
String method = request.getMethod();
//2. 根据请求方式再调用对应的处理方法
if("GET".equalsIgnoreCase(method)){
doGet(request, response);
}else if("POST".equalsIgnoreCase(method)){
doPost(request, response);
}
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException{
// TODO Auto-generated method stub
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// TODO Auto-generated method stub
}
3). 实际开发中, 直接继承 HttpServlet, 并根据请求方式复写 doXxx() 方法即可.
4). 好处: 直接由针对性的覆盖 doXxx() 方法; 直接使用 HttpServletRequest 和 HttpServletResponse, 不再需要强转.
1). 是一个 Servlet, 继承自 GenericServlet. 针对于 HTTP 协议所定制.
2). 在 service() 方法中直接把 ServletReuqest 和 ServletResponse 转为 HttpServletRequest 和 HttpServletResponse.
并调用了重载的 service(HttpServletRequest, HttpServletResponse)
在 service(HttpServletRequest, HttpServletResponse) 获取了请求方式: request.getMethod(). 根据请求方式有创建了
doXxx() 方法(xxx 为具体的请求方式, 比如 doGet, doPost)
@Override
public void service(ServletRequest req, ServletResponse res)
throws ServletException, IOException {
HttpServletRequest request;
HttpServletResponse response;
try {
request = (HttpServletRequest) req;
response = (HttpServletResponse) res;
} catch (ClassCastException e) {
throw new ServletException("non-HTTP request or response");
}
service(request, response);
}
public void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
//1. 获取请求方式.
String method = request.getMethod();
//2. 根据请求方式再调用对应的处理方法
if("GET".equalsIgnoreCase(method)){
doGet(request, response);
}else if("POST".equalsIgnoreCase(method)){
doPost(request, response);
}
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException{
// TODO Auto-generated method stub
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// TODO Auto-generated method stub
}
3). 实际开发中, 直接继承 HttpServlet, 并根据请求方式复写 doXxx() 方法即可.
4). 好处: 直接由针对性的覆盖 doXxx() 方法; 直接使用 HttpServletRequest 和 HttpServletResponse, 不再需要强转.