servlet
这部分结合之前的几章一起看,以免遗漏。http://blog.163.com/qqabc20082006@126/blog/static/2292852520106249123739/Servlet什么是Servlet Servlet是一个JavaEE组件,是在服务器端运行以处理客户端请求并作出响应的程序。Servlet何时被创建
这部分结合之前的几章一起看,以免遗漏。
http://blog.163.com/qqabc20082006@126/blog/static/2292852520106249123739/
Servlet
什么是Servlet
Servlet是一个JavaEE组件,是在服务器端运行以处理客户端请求并作出响应的程序。
Servlet何时被创建
1,默认情况下,当WEB客户第一次请求访问某个Servlet的时候,WEB容器将创建这个Servlet的实例。
2,当web.xml文件中如果<servlet>元素中指定了<load-on-startup>子元素时,Servlet容器在启动web服务器时,将按照顺序创建并初始化Servlet对象。
注意:在web.xml文件中,某些Servlet只有<serlvet>元素,没有<servlet-mapping>元素,这样我们无法通过url的方式访问这些Servlet,这种Servlet通常会在<servlet>元素中配置一个<load-on-startup>子元素,让容器在启动的时候自动加载这些Servlet并调用init()方法,完成一些全局性的初始化工作。
http://www.cnblogs.com/cuiliang/archive/2011/10/21/2220671.html
Servlet的执行过程
首先,客户端发送请求到服务器端;
其次,服务器端根据web.xml文件中的Servlet相关配置信息,将客户端请求转发到相应的Servlet;
之后,Servlet会根据request对象中封装的用户请求与数据库进行交互,返回数据之后,Servlet会将返回的数据封装到response对象中;
此时,控制权从Servlet重新回到服务器端,最后,服务器端将响应信息返回给客户端,并且跳转到相应的页面。
Servlet生命周期分为三个阶段:
1,初始化阶段 调用init()方法
2,响应客户请求阶段 调用service()方法
3,终止阶段 调用destroy()方法
Servlet初始化阶段
在下列时刻Servlet容器装载Servlet:
1,Servlet容器启动时自动装载某些Servlet,实现它只需要在web.XML文件中的<Servlet></Servlet>之间添加如下代码:
<loadon-startup>
1
</loadon-startup>
2,在Servlet容器启动后,客户首次向Servlet发送请求
3,Servlet类文件被更新后,重新装载Servlet
Servlet被装载后,Servlet容器创建一个Servlet实例并且调用Servlet的init()方法进行初始化。
在Servlet的整个生命周期内,init()方法只被调用一次。
Servlet响应请求阶段
对于用户到达Servlet的请求,Servlet容器会创建特定于这个请求的ServletRequest对象和ServletResponse对象,然后调用Servlet的service方法。service方法从ServletRequest对象获得客户请求信息,处理该请求,并通过ServletResponse对象向客户返回响应信息。
对于Tomcat来说,它会将传递过来的参数放在一个Hashtable中,该Hashtable的定义是:
private
Hashtable<String String[]> paramHashStringArray =
new
Hashtable<String String[]>();
|
这是一个String-->String[]的键值映射。
HashMap线程不安全的,Hashtable线程安全。
-----------------------------------------------------------------------------------------------------------------------------------
Servlet终止阶段
当WEB应用被终止,或Servlet容器终止运行,或Servlet容器重新装载Servlet新实例时,Servlet容器会先调用Servlet的destroy()方法,在destroy()方法中可以释放掉Servlet所占用的资源。
Servlet类的继承关系
客户发送一个请求,Servlet是调用service()方法对请求进行响应的。
通过源代码可见,service()方法中对请求的方式进行了匹配,选择调用doGet,doPost等这些方法,然后再进入对应的方法中调用逻辑层的方法,实现对客户的响应。
在Servlet接口和GenericServlet中是没有doGet,doPost等等这些方法的,HttpServlet中定义了这些方法,但是都是返回error信息,所以,我们每次定义一个Servlet的时候,都必须实现doGet或doPost等这些方法。
每一个自定义的Servlet都必须实现Servlet的接口,Servlet接口中定义了五个方法,其中比较重要的三个方法涉及到Servlet的生命周期,分别是上文提到的init(),service(),destroy()方法。
GenericServlet是一个通用的,不特定于任何协议的Servlet,它实现了Servlet接口。
而HttpServlet继承于GenericServlet,因此HttpServlet也实现了Servlet接口。所以我们定义Servlet的时候只需要继承HttpServlet即可。
Servlet接口和GenericServlet是不特定于任何协议的,而HttpServlet是特定于HTTP协议的类,所以HttpServlet中实现了service()方法,并将请求ServletRequest,ServletResponse强转为HttpRequest和HttpResponse。
public
void
service(ServletRequest req,ServletResponse res)
throws
ServletException,IOException
{
HttpRequest request;
HttpResponse response;
try
{
req = (HttpRequest)request;
res = (HttpResponse)response;
}
catch
(ClassCastException e)
{
throw
new
ServletException(
"non-HTTP request response"
);
}
service(request,response);
}
|
代码的最后调用了HTTPServlet自己的service(request,response)方法,然后根据请求去调用对应的doXXX方法,因为HttpServlet中的doXXX方法都是返回错误信息,
protected
void
doGet(HttpServletRequest res,HttpServletResponse resp)
throws
ServletException,IOException
{
String protocol = req.getProtocol();
String msg = IStrings.getString(
"http.method_get_not_supported"
);
if
(protocol.equals(
"1.1"
))
{
resp.sendError(HttpServletResponse.SC.METHOD.NOT.ALLOWED,msg);
}
esle
{
resp.sendError(HttpServletResponse.SC_BAD_REQUEST,msg);
}
}
|
所以需要我们在自定义的Servlet中override这些方法!
Servlet的实现:Servlet接口->GenericServlet类->HttpServlet类->用户自定义的Servlet。
Servlet与JSP的关系及区别
Servlet是JSP的基础,JSP是Servlet技术的扩展。JSP运行之前首先将编译为一个Servlet。
JSP侧重于视图;Servlet主要用于控制业务逻辑。
Servlet API中的forward()与redirect()的区别?
请求转发:forward是Request对象的方法,它是在服务器端执行,并且始终在同一个Request域中,所以页面间可以共享Request对象中的资源。转发后,客户端浏览器地址不会改变;请求转发性能优于重定向;
<jsp:forward>
servlet中的实现:
RequestDispatcher rd = request.getRequestDispatcher("login2");
rd.forward(request, response);-----------------服务器端将request和response对象直接转发给目的servlet处理。reques不变。
重定向:redirect是response对象的方法,发生在客户端浏览器,它有两次Request请求,第二次请求将丢失第一次Request中的资源。重定向之后,客户端浏览器地址发生改变。
servlet中的实现:response.sendRedirect("error.jsp");----------------响应发给客户端,客户端再向"error.jsp"发出请求。request改变。
Servlet API中的include()
<jsp:include>
servlet中的实现:
RequestDispatcher rd = request.getRequestDispatcher("login2");
rd.include(request, response);-----------------服务器端将request和response对象直接交给被包含的servlet处理。reques不变。
Servlet是单实例的,但有两种访问模式:多线程以及单线程访问,前者会有线程安全方面的问题,后者有请求排队等待的问题。
Servlet如何同时处理多个请求?
Servlet采用多线程来处理多个请求的同时访问。Servlet容器通过线程池来管理维护服务请求。所谓线程池,相当于数据库连接池,实际上是等待执行代码的一组线程,叫做工作者线程。Servlet容器通过一个调度线程来管理工作者线程。
· 当容器收到一个Servlet的访问请求,调度者线程就从线程池中选出一个工作者线程,将用户请求传递给该线程,然后由该线程处理Servlet的service()方法;
· 当这个线程在执行的时候,容器收到一个新的请求,调度者线程再次从线程池中选出一个新的工作者线程;
· 当容器同时收到对同一个Servlet的多个请求时,那么Servlet的service方法将在多线程中并发执行。
注:1.Servlet容器默认采用单实例多线程的方式来处理请求。这样减少了产生Sevlet实例的开销,提升了对请求的响应时间;
2.对于Tomcat容器来讲,可以在其server.xml中通过<Connector>中设置线程池中的线程数目。
Servlet也可以采用单线程模式!
如何开发线程安全的Servlet?
Servlet容器采用多线程来处理请求,提高性能的同时也造成了线程安全问题。要开发线程安全的Servlet应该从一下几个方面进行:
1. 变量的线程安全; 多线程并不共享局部变量,所以我们要尽可能的在Servlet中使用局部变量;
2. 代码块的线程安全; 使用同步块Synchronized,防止可能调用的代码块;但是要注意的是,要尽可能得缩小同步代码的方范围,不要在service方法和响应方法上直接使用同步,这会严重影响性能。
3. 属性的线程安全;ServletContext,HttpSession,ServletRequest对象中属性;
4. 使用同步集合; 使用Vector代替ArrayList,使用HashTable代替HashMap;
5. 不要在Servlet中创建自己的线程来完成某个功能; Servlet本身就是多线程的,如果再创建新的线程,将会导致线程执行复杂化,出现线程安全问题;
6. 在多个Servlet中,对外部对象,比如:文件;进行修改操作一定要加锁,做到互斥访问;
如何实现servlet的单线程模式?
javax.servlet.SingleThreadModel接口是一个标识接口,如果一个Servlet实现了这个接口,那Servlet容器将保证在一个时刻仅有一个线程可以在给定的servlet实例的service方法中执行,将其他所有请求进行排队。实现方法:<%@ page isThreadSafe="false" %>
更多推荐
所有评论(0)