这部分结合之前的几章一起看,以免遗漏。

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" %>


Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐