原文: https://howtodoinjava.com/tomcat/a-birds-eye-view-on-how-web-servers-work/

许多时候,我们想知道 Web 容器 / Web 服务器(例如 tomcat 或 jboss)如何工作? 他们如何处理来自世界各地的传入 HTTP 请求? 幕后发生的原因是什么? Java Servlet API(例如ServletContextServletRequestServletResponseSession之类的类)如何适应图片? 这些是非常重要的问题/概念,如果您是 Web 应用开发人员或渴望成为,则必须知道。 在这篇文章中,我将尝试找出上述问题的答案,即使不是全部。 从这里保持集中。

  1. Table of Contents:
  2. What are web server, application server and web container?
  3. What are Servlets? How they help?
  4. What is ServletContext? Who creates it?
  5. Where ServletRequest and ServletResponse fits into life cycle?
  6. How Session is managed? Know the cookie?
  7. How thread safety should be ensured?

什么是 Web 服务器,应用服务器和 Web 容器?

我将首先讨论 Web 服务器和应用服务器。 我说一句

“从历史上看,它们是不同的,但是这两个以前截然不同的类别逐渐合并,现在在大多数情况和用途中应被视为一个实体。”

Mosaic 浏览器(通常被称为第一个图形化 Web 浏览器)和超链接内容的早期,出现了一种新概念“Web 服务器”,用于服务静态网页内容和 HTTP 协议上的图像。 很简单。 如今,大多数内容都是静态的,HTTP 1.0 协议只是一种随身携带文件的方式。 但是很快,Web 服务器就发展为具有 CGI 功能。 这意味着在每个 Web 请求上有效启动一个流程以生成动态内容。 到那时,HTTP 协议也已经成熟,Web 服务器变得更加复杂,并具有诸如缓存,安全性和会话管理之类的附加功能。 随着技术的进一步成熟,我们从 Kiva 和 NetDynamics 中获得了基于公司的基于 Java 的服务器端技术,最终它们全部合并为 JSP(java 服务器页面), 在当今大多数应用开发中使用。

Web 服务器如何工作? - 图1

这是关于网络服务器的。 现在,我们来讨论应用服务器

在并行类别中,应用服务器已经发展并存在了很长时间。 一些公司为 Unix 提供了诸如 Tuxedo(面向事务的中间件),TopEndEncina 等产品,这些产品从大型机应用管理中衍生而来 和监视环境,例如 IMS 和 CICS。 这些产品大多数都指定了“封闭式”产品特定的通信协议,以将“胖”客户端互连到服务器。 在 90 年代,这些传统的应用服务器产品首先通过网关开始嵌入基本的 HTTP 通信功能。 很快,这两类之间的界限开始模糊。

到那时,Web 服务器在处理更高的负载,更多的并发和更好的功能方面变得越来越成熟。 应用服务器开始提供越来越多的基于 HTTP 的通信功能。 所有这些都导致 Web 服务器和应用服务器之间的界限很窄。

在这一点上,“应用服务器”和“网络服务器”之间的界线是模糊的。 但是,作为重点,人们继续使用不同的术语。

当有人说“Web 服务器”时,您通常会想到以 HTTP 为中心,面向 Web UI 的应用。 当有人说“应用服务器”时,您可能会认为“负载较重,企业功能,事务和排队,多通道通信(HTTP 及其他)”。 但是大多数情况下,这是同一款产品,如今可以同时满足这两种要求。

这就是有关 Web 服务器和应用服务器的一切。 现在进入第三个术语,即网络容器。

Web 服务器如何工作? - 图2

Web 容器,尤其是 java 中的,应引用 servlet 容器。 Servlet 容器是与 Java Servlet 交互的 Web 服务器的组件。 Web 容器负责管理 Servlet 的生命周期,将 URL 映射到特定的 Servlet,并确保 URL 请求者具有正确的访问权限以及更多此类服务。 基本上,综合以上所有事实, Servlet 容器是 Servlet 运行并维持其生命周期的运行时环境。

什么是 Servlet? 他们如何提供帮助?

在 Java 中,servlet 使您可以编写服务器端组件,以根据请求帮助生成动态内容。 实际上,Servlet 是在javax.servlet包中定义的接口。 它声明了 Servlet 生命周期的三种基本方法 - init()service()destroy()。 它们由每个 servlet(在 SDK 中定义或由用户定义)实现,并在服务器的生命周期中由服务器在特定时间调用。

Servlet 类由其类加载器通过延迟加载或急切加载动态地加载到容器中。 每个请求都在其自己的线程中,并且 servlet 对象可以同时服务多个线程。 当不再使用它时,它将被 JVM 垃圾回收。

延迟加载 servlet

Web 服务器如何工作? - 图3 Web 服务器如何工作? - 图4

立即加载 Servlet

Web 服务器如何工作? - 图5 Web 服务器如何工作? - 图6

什么是ServletContext? 谁创造的?

当 Servlet 容器启动时,它将部署并加载所有 Web 应用。 加载 Web 应用后,Servlet 容器将为每个应用创建一次ServletContext,并将其保留在服务器的内存中。 该 Web 应用的web.xml将被解析,并且在web.xml中找到的每个 Servlet,过滤器和监听器都将被创建一次并保存在服务器的内存中。 当 Servlet 容器关闭时,它将卸载所有 Web 应用,并且ServletContext以及所有 Servlet,过滤器和监听器实例都将被丢弃。

根据 Java 文档, ServletContext定义了 Servlet 用于与其 Servlet 容器通信的一组方法,例如,获取文件的 MIME 类型,调度请求或写入日志文件。 对于 Web 应用在其部署描述符中标记为“已分发”的情况,每个虚拟机都有一个上下文实例。 在这种情况下,上下文不能用作共享全局信息的位置(因为该信息不是真正的全局信息)。 请改用外部资源,例如数据库。

ServletRequestServletResponse适合生命周期的地方?

Servlet 容器连接到 Web 服务器,该服务器在特定端口号(通常为 80)上监听 HTTP 请求。当客户端(具有 Web 浏览器的用户)发送 HTTP 请求时, Servlet 容器将创建新的HttpServletRequestHttpServletResponse对象将其传递给 URL 样式与请求 URL 匹配的已创建过滤器和 Servlet 实例的方法,所有这些都在同一线程中。

请求对象提供对 HTTP 请求的所有信息的访问,例如请求标头和请求正文。 响应对象提供了以所需方式控制和发送 HTTP 响应的功能,例如设置标头和正文(通常带有 JSP 文件中的 HTML 内容)。 提交并完成 HTTP 响应后,请求和响应对象都将被丢弃。

如何管理会话? 知道 cookie 吗?

当客户端首次访问 Web 应用和/或要通过request.getSession()首次获取HttpSession时,则 Servlet 容器将创建它,并生成一个长而唯一的 ID(您可以通过session.getId()获取并存储在服务器的内存中。 servlet 容器还将在 HTTP 响应中设置 Cookie,其中JSESSIONID为 cookie 名称,唯一会话 ID 为 cookie 值。

根据 HTTP cookie 规范(体面的 Web 浏览器和 Web 服务器必须遵守的约定),要求客户端(Web 浏览器)在后续请求中将该 Cookie 发送回 只要 Cookie 有效即可。 Servlet 容器将确定每个传入的 HTTP 请求标头中是否存在名为JSESSIONID的 cookie,并使用其值从服务器的内存中获取关联的HttpSession

HttpSession会一直存在,直到没有被使用为止,您可以在web.xml中指定该设置,该设置默认为 30 分钟。 因此,如果客户端超过 30 分钟不再访问该 Web 应用,则 Servlet 容器将破坏该会话。 即使指定了 cookie,每个后续请求都将无法再访问同一会话。 servlet 容器将创建一个新的。

现有会话

Web 服务器如何工作? - 图7

新会话

Web 服务器如何工作? - 图8

另一方面,客户端上的会话 cookie 具有默认生存期,该生存期只要浏览器实例正在运行即可。 因此,当客户端关闭浏览器实例(所有选项卡/窗口)时,会话将在客户端被丢弃。 在新的浏览器实例中,与该会话关联的 cookie 将不再发送。 一个新的request.getSession()将返回一个全新的HttpSession并设置一个具有全新会话 ID 的 cookie。

应如何确保线程安全?

您现在应该已经了解到,Servlet 和过滤器在所有请求之间共享。 这是 Java 的优点,因为它是多线程,并且不同的线程(即 HTTP 请求)可以使用同一实例。 否则,在每个请求上重新创建它会太昂贵。

Web 服务器如何工作? - 图9

但是,您还应该意识到,永远不要将任何请求或会话范围的数据分配为 Servlet 或过滤器的实例变量。 它将在其他会话中的所有其他请求之间共享。 那是线程不安全的! 下面的示例说明:

  1. public class MyServlet extends HttpServlet
  2. {
  3. private Object thisIsNOTThreadSafe; //Don't to this
  4. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
  5. {
  6. Object thisIsThreadSafe;
  7. thisIsNOTThreadSafe = request.getParameter("foo"); // BAD!! Shared among all requests!
  8. thisIsThreadSafe = request.getParameter("foo"); // OK, this is thread safe.
  9. }
  10. }

不要这样做。 这将导致软件错误。

以上就是这个主题。 请继续关注更多此类帖子。 最好订阅电子邮件新闻信,以在收件箱中获得通知。

祝您学习愉快!

推荐链接

http://www.invir.com/int-prog-cgi.html