servlet介绍

 
  • Servlet概述:

      (Server + Applet =Servlet) 意为服务器端的小程序.他具有独立于平台和协议的特性,可以生成动态的web页面。他担当客户请求(Web浏览器或其他HTTP客户程序)与服务器响应(HTTP服务器上的数据库或应用程序)的中间层。Servlet是位于web服务器内部的服务器端的java应用程序,与传统的从命令行启动的java应用程序不同。Servlet由web服务器进行加载,改web服务器必须包含支持servlet的java虚拟机。

       服务器上需要一些程序,常常是根据用户输入访问数据库的程序。这些通常是使用公共网关接口(CGI Common Gateway Interface)应用程序完成的。然而,在服务器上运行java,这种程序可使用java编程语言实现。在通信量大的服务器上,java servlet的优点在于他们的执行速度更快于CGI程序。各个用户请求被激活成单个程序中的一个线程,而无需创建单独的进程,这意味着服务器端处理请求系统开销将明显降低。

       Servlet的主要功能在于交互式的浏览和修改数据,生成动态web内容。

       同时,servlet提供了大量的使用工具例程,例如自动的解析HTML表单数据、读取和设置HTTP头,处理Cookie、跟踪会话状态等。

       Servlet能够在各个程序之间共享数据,使得数据库连接池之类的功能很容易实现。

      

  • Servlet工作模式

       客户端发送请求至服务器。

       服务器启动并调用servlet,servlet根据客户端请求生成响应内容并将其传给服务器。

       服务器将响应返回客户端

  • Servlet如何体现的模板方法模式

       我们知道servlet都是继承自HttpServlet的。而执行servlet实际上执行的就是service方法。在这个方法中,采用一定的逻辑决定什么时候调用doGet,doPost方法。而doPost,doGet方法在这个方法中有默认实现。但在继承的子类中可以覆盖。

       在这里,模板中并不是严格上用的虚方法,在子类中去实现。但是,他的精髓还是模板方法的精髓。这也说明,我们在应用的过程中,不要生硬的去模板,要能适当的根据实际情况去进行变化。

  • Servlet生命周期

        当一个请求映射到一个servlet时,该容器执行下列步骤:

Server创建一个servlet实例

Server调用servlet的init()方法

一个客户端的请求到达server

Server创建一个请求对象

Server创建一个相应对象

Server激活servlet的service()方法,传递请求和相应对象作为参数

Service()方法获得关于请求对象的信息,处理请求,访问其他资源,获得需要的信息。

Service()方法使用相应对象的方法,将响应传回server,最终到达客户端。Service()方法可能激活其他方法以处理请求,如doGet()或doPost()或程序员自己开发的新的方法。

 

       对于更多的客户端请求,server创建心的请求和相应对象,仍然激活此servlet的service()方法,将这两个对象作为参数传递给他。如此重复以上的循环,但无需再次调用init()方法。一般servlet只初始化一次(只有一个对象),当server不再需要servlet时(一般当server关闭时),server调用servlet的destroy()方法。

      

  • HTTPServlet应用编程接口

  HTTP Servlet 使用一个 HTML 表单来发送和接收数据。要创建一个 HTTP Servlet,请扩展 HttpServlet 类, 该类是用专门的方法来处理 HTML 表单的 GenericServlet 的一个子类。 HTML 表单是由 <FORM> 和 </FORM> 标记定义的。表单中典型地包含输入字段(如文本输入字段、复选框、单选按钮和选择列表)和用于提交数据的按钮。当提交信息时,它们还指定服务器应执行哪一个Servlet(或其它的程序)。 HttpServlet 类包含 init()、destroy()、service() 等方法。其中 init() 和 destroy() 方法是继承的。

       (1) init() 方法

  在 Servlet 的生命期中,仅执行一次 init() 方法。它是在服务器装入 Servlet 时执行的。 可以配置服务器,以在启动服务器或客户机首次访问 Servlet 时装入 Servlet。 无论有多少客户机访问 Servlet,都不会重复执行 init() 。

  缺省的 init() 方法通常是符合要求的,但也可以用定制 init() 方法来覆盖它,典型的是管理服务器端资源。 例如,可能编写一个定制 init() 来只用于一次装入 GIF 图像,改进 Servlet 返回 GIF 图像和含有多个客户机请求的性能。另一个示例是初始化数据库连接。缺省的 init() 方法设置了 Servlet 的初始化参数,并用它的 ServletConfig 对象参数来启动配置, 因此所有覆盖 init() 方法的 Servlet 应调用 super.init() 以确保仍然执行这些任务。在调用 service() 方法之前,应确保已完成了 init() 方法。

  (2) service() 方法

  service() 方法是 Servlet 的核心。每当一个客户请求一个HttpServlet 对象,该对象的service() 方法就要被调用,而且传递给这个方法一个"请求"(ServletRequest)对象和一个"响应"(ServletResponse)对象作为参数。在 HttpServlet 中已存在 service() 方法。缺省的服务功能是调用与 HTTP 请求的方法相应的 do 功能。例如, 如果 HTTP 请求方法为 GET,则缺省情况下就调用 doGet() 。Servlet 应该为 Servlet 支持的 HTTP 方法覆盖 do 功能。因为 HttpServlet.service() 方法会检查请求方法是否调用了适当的处理方法,不必要覆盖 service() 方法。只需覆盖相应的 do 方法就可以了。

  Servlet的响应可以是下列几种类型:

  一个输出流,浏览器根据它的内容类型(如text/HTML)进行解释。

  一个HTTP错误响应, 重定向到另一个URL、servlet、JSP。

  (3)doGet()方法

  当一个客户通过HTML 表单发出一个HTTP GET请求或直接请求一个URL时,doGet()方法被调用。与GET请求相关的参数添加到URL的后面,并与这个请求一起发送。当不会修改服务器端的数据时,应该使用doGet()方法。

  (4)doPost()方法

  当一个客户通过HTML 表单发出一个HTTP POST请求时,doPost()方法被调用。与POST请求相关的参数作为一个单独的HTTP 请求从浏览器发送到服务器。当需要修改服务器端的数据时,应该使用doPost()方法。

  (5) destroy() 方法

  destroy() 方法仅执行一次,即在服务器停止且卸装Servlet 时执行该方法。典型的,将 Servlet 作为服务器进程的一部分来关闭。缺省的 destroy() 方法通常是符合要求的,但也可以覆盖它,典型的是管理服务器端资源。例如,如果 Servlet 在运行时会累计统计数据,则可以编写一个 destroy() 方法,该方法用于在未装入 Servlet 时将统计数字保存在文件中。另一个示例是关闭数据库连接。

  当服务器卸装 Servlet 时,将在所有 service() 方法调用完成后,或在指定的时间间隔过后调用 destroy() 方法。一个Servlet 在运行service() 方法时可能会产生其它的线程,因此请确认在调用 destroy() 方法时,这些线程已终止或完成。

  • Servlet线程安全问题

         Servlet体系结构是建立在java多线程机制之上的,他的生命周期是由web容器负责的。当客户端第一次请求某个servlet时,servlet容器将会根据web.xml配置文件实例化这个servlet类。当有新的客户端请求该servlet时,一般不用再实例化改servlet类,也就是有多个线程在使用这个实例。Servlet容器会自动使用线程池等技术来支持系统的运行。如

         这样,当两个或多个线程同时访问同一个servlet时,可能会发生多个线程同时访问同一个资源的情况,数据可能会变得不一致。

        避免servlet线程安全问题的一般情况下有三种方式。

1、  实现 SingleThreadModel 接口

public class Concurrent Test extends HttpServlet implements SingleThreadModel {
…………
}

2、 同步对共享数据的操作

…………
Public class Concurrent Test extends HttpServlet {
    …………
username = request.getParameter ("username");
synchronized (this){
output = response.getWriter ();
try {
    Thread. Sleep (5000);
} Catch (Interrupted Exception e){}
    output.println("用户名:"+Username+"
");
}
}
}

3、 避免使用实例变量

对上面的三种方法进行测试,可以表明用它们都能设计出线程安全的Servlet程序。但是,如果一个Servlet实现了 SingleThreadModel接口,Servlet引擎将为每个新的请求创建一个单独的Servlet实例,这将引起大量的系统开销。 SingleThreadModel在Servlet2.4中已不再提倡使用;同样如果在程序中使用同步来保护要使用的共享的数据,也会使系统的性能大大下降。这是因为被同步的代码块在同一时刻只能有一个线程执行它,使得其同时处理客户请求的吞吐量降低,而且很多客户处于阻塞状态。另外为保证主存内容和线程的工作内存中的数据的一致性,要频繁地刷新缓存,这也会大大地影响系统的性能。所以在实际的开发中也应避免或最小化 Servlet 中的同步代码;在Serlet中避免使用实例变量是保证Servlet线程安全的最佳选择

        Servlet的线程安全问题只有在大量的并发访问时,才会显现出来,并且很难发现,因此在编写servlet时要特别注意。

 

        参考博文:http://www.sf.org.cn/j2me/base/20922.html

                         http://www.blogjava.net/menlong999/archive/2008/09/17/229332.html 

转载于:https://www.cnblogs.com/yanmei-yaomy/archive/2012/02/23/3024798.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值