Tomcat中的零停机部署(和回滚); 演练和清单

亲爱的大家,

如果您认为Tomcat不能再进步,那您就错了。 Tomcat 7引入了所谓的并行部署 。 这是由SpringSource / VMWare贡献的。

简而言之,并行部署是一种能够并行部署一个以上版本的Web应用程序的功能,使所有版本都可以在完全相同的URL下使用。

考虑一下。 如果您有新版本的应用程序,则只需将其放入运行旧版本的Tomcat中 ,它将运行。 实际上,它们都会起作用。 Tomcat处理应用程序版本之间的所有会话管理和流量路由。 无需重新启动Tomcat 。 无需停止处理请求。 无需与您的老板谈论停机时间。 您的老板无需与任何客户谈论停机时间。

让我们看看实际情况,对吧? 使用下面的命令,您可以启动一个最小的Web应用程序来演示此功能。

$ mkdir WEB-INF
$ echo "" > WEB-INF/web.xml
$ echo 'old version ' > index.jsp
$ jar cf foo##001.war WEB-INF index.jsp
$ echo 'NEW version ' > index.jsp
$ jar cf foo##002.war WEB-INF index.jsp

现在,您有两个名为foo ## 001.warfoo ## 002.war的 Web应用程序。 ## 001## 002指定WAR文件的版本号。 每个都有自己的索引页面,该页面显示当前时间以及是旧的还是新的Web应用程序。 创建此功能的人员选择了一个非常简单的解决方案,用于告诉Tomcat什么是替代版本。 您要做的就是在WAR的文件名上加上## <version> 。 简单有效,如果看起来有些古怪。

现在部署Web应用程序的“旧”版本。

$ cp foo##001.war apache-tomcat-7.0.12/webapps/

打开浏览器,输入WAR文件的URL(例如http:// localhost:8080 / foo )并注意时间。 请注意,您在URL上看不到版本号。 页面每秒自动刷新。 在表面之下, Tomcat将与您的浏览器建立会话。 以后再说。

现在部署Web应用程序的“新”版本。

$ cp foo##002.war apache-tomcat-7.0.12/webapps/

请注意,在已经打开的浏览器窗口中,时间仍在滴答作响,并且仍显示旧版本。 打开第二个浏览器,在该浏览器中,也打开http:// localhost:8080 / foo 。 为了获得最佳结果,请使用完全不同的浏览器以避免任何会话异常。 我使用Safari和Opera进行了测试。

您应该看到第二个浏览器选择了新的Web应用程序,而旧的Web应用程序仍在使用第一个浏览器。

很整洁吧?

好的,您是否破坏了部署并想回滚? 很简单,只需删除新版本, Tomcat就会自动退回到使用旧版本。 辛苦了 现在就试试:

$ rm apache-tomcat-7.0.12/webapps/foo##002.war

您会注意到网页会自动切换为使用旧版本的应用程序。

您将必须制定自己的部署策略。 您可能选择让旧版本耗尽。 在旧应用程序上的所有会话都到期后,您可以从Tomcat中删除旧部署。 另一方面,您可以只保留旧代码。 它不会造成任何伤害。

当您想开始在Tomcat服务器上使用版本化的WAR文件时,需要考虑一些事项。 因此,在您离开并更改公司的部署策略之前,请检查以下列表。

  1. 内部缓存应该是直写的,并且很快过期
  2. 您需要启用会话
  3. 日志去哪儿了?
  4. 磁盘文件和目录需要共享
  5. 没有TCP套接字监听器
  6. 您的应用必须能够取消部署

我将按顺序解释这些内容。 大多数是主题的变体; 考虑一下您的代码对机器资源所做的假设。

内部缓存应该是直写的,并且很快过期

同一Web应用程序的不同版本各自具有自己的类上下文。 这意味着需要检查Web应用程序中的所有本地缓存​​。 如果您主动缓存并长时间保留缓存的信息,则该Web应用程序的一个版本可能看不到另一个版本所做的更改。

设想一个情况,其中两个版本的Web应用程序都使用相同的数据库,并且都具有本地缓存​​以避免访问该数据库。 如果该Web应用程序的一个版本更改了数据库中的记录,则其他版本将看不到该更改,直到该信息自己的缓存版本过期为止。

如果您在Web应用程序中执行任何内存中缓存,请在确定缓存不会提供不可接受的过时信息之前不要使用并行部署。

您需要启用会话

Tomcat使用其自己的会话管理来确定应由哪个版本的Web应用程序处理哪些请求。 如果您打算自己实现会话处理,或者如果您在Tomcat中关闭了会话处理,那么并行部署将对您不起作用。

日志去哪儿了?

您可能指定将日志记录写入某个地方的日志文件。 如果未在日志文件名中使用应用程序的完整上下文名称,则可能会遇到两种情况:Web应用程序的两个版本都写入同一日志文件。 这样做的问题是,您可能不知道应用程序的哪个版本生成了在该文件中找到的输出。

磁盘文件和目录需要共享

Java EE设计者的意图一直是使Web应用程序独立于底层机器和文件系统。 如果您的应用程序使用数据文件,请花点时间考虑一下当您的Web应用程序的两个版本开始读写它们时会发生什么。

特别是,请考虑将Java的监视器和锁限制在单个上下文中。 因此,如果您使用某种类型的锁来保护对文件的访问,则拥有该Web应用程序的两个版本意味着您在JVM中具有两个锁,从而可能允许两个线程对其进行访问。

没有TCP套接字监听器

一些应用程序不仅仅服务于HTTP请求。 他们有自己的TCP套接字处理程序为客户端提供服务。 通过部署一个以上版本的Web应用程序,您可以获得多个侦听器。 显然这是行不通的。 只有一个侦听器可以在任何给定端口上侦听。

您的应用必须能够取消部署

如果您希望能够回退已损坏的发行版,或者希望清理Web应用程序的旧的,未使用的版本,则需要彻底取消部署Web应用程序。 幸运的是, Tomcat也可以帮助您。

希望这些警告不会使您停止使用Tomcat中的零停机时间部署。 您可能需要解决一些问题才能使其正常运行。 但是,让我们面对现实吧,通过使您的应用程序对资源的假设更少,无论如何您将获得一个更强大的应用程序。

PS。 我喜欢将SVN修订号用作WAR版本命名方案。 因此,我的WAR文件被命名为foo ## <svn版本=””>。war 。 唯一需要注意的是,将版本作为字符串进行比较以确定版本顺序。 因此,您可能必须对版本号进行零填充以确保正确订购。

参考: Tomcat中的零停机部署(和回滚); JCG合作伙伴 Kees Jan在Java Monitor论坛上 的演练和清单

快乐编码
拜伦

相关文章:


翻译自: https://www.javacodegeeks.com/2011/06/zero-downtime-deployment-and-rollback.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/374880.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javaweb 学习资源

http://jinnianshilongnian.iteye.com/category/231099转载于:https://www.cnblogs.com/sishahu/p/5368018.html

HDU 1863 畅通工程(最小生成树,prim)

题意&#xff1a; 给出图的边和点数&#xff0c;要求最小生成树的代价&#xff0c;注&#xff1a;有些点之间是不可达的&#xff0c;也就是可能有多个连通图。比如4个点&#xff0c;2条边:1-2&#xff0c;3-4。 思路&#xff1a; 如果不能连通所有的点&#xff0c;就输出‘?’…

2000年不算在21世纪

练习3-5 输出闰年 (15 分) 输出21世纪中截止某个年份以来的所有闰年年份。注意&#xff1a;闰年的判别条件是该年年份能被4整除但不能被100整除、或者能被400整除。 想当然地以为21世纪是2000~2099&#xff0c;当然没有通过 if(N > 2000&&N < 2099){for(int i …

使用迭代器时如何避免ConcurrentModificationException

Java Collection类是快速失败的&#xff0c;这意味着如果在使用迭代器遍历某个线程的同时更改了Collection&#xff0c;则iterator.next&#xff08;&#xff09;将抛出ConcurrentModificationException 。 在多线程以及单线程环境下都可能出现这种情况。 让我们通过以下示例探…

Sublime Text 3实用快捷键大全

下面是我通过网上教程和文本资料学习sublime Text3时收集的一些实用功能和常用快捷键&#xff0c;现在分享出来&#xff0c;如果还有其它的好用的功能可以在下面留言&#xff0c;以便互相学习和交流&#xff0c;谢谢&#xff01;。 选择类 CtrlD 选中光标所占的文本&#xff0c…

Tomcat中配置JNDI数据源

准备工作&#xff1a; Tomcat版本&#xff1a;tomcat6.0以上 下例中均使用MySQL数据库 将对应数据源的jar包和MySQL的驱动包拷贝至tomcat的lib文件夹下 一、全局数据源 1步骤一&#xff1a;配置 在tomcat下的conf/server.xml的GlobalNamingResources节点标签中增加如下配置&…

练习3-8 查询水果价格 (15 分)

练习3-8 查询水果价格 (15 分) 给定四种水果&#xff0c;分别是苹果&#xff08;apple&#xff09;、梨&#xff08;pear&#xff09;、桔子&#xff08;orange&#xff09;、葡萄&#xff08;grape&#xff09;&#xff0c;单价分别对应为3.00元/公斤、2.50元/公斤、4.10元/公…

JavaFX 2.0 beta示例应用程序和思考

我有一段时间回过头来玩JavaFX&#xff0c;并且在使用该语言方面有好有坏的经验。 随着JavaFX 2.0 beta的发布&#xff0c;我想尝试一下。 在这里&#xff0c;我开发了一个简单的地址解析应用程序&#xff0c;该应用程序将使用Google地址编码API来获取地址并提供该位置的纬度-经…

$Android自定义控件在不同状态下的属性

在写代码的时候&#xff0c;有时候需要控件在不同状态下显示不同的外观&#xff0c;比如在按钮按下的时候要变颜色&#xff0c;EditText获取焦点时候边框要变颜色等。那么下面就来梳理一下这些是怎么实现的。 &#xff08;一&#xff09;按钮按下时候变颜色 1、在项目的drawabl…

解析DBR操作系统引导记录数据

理解文件系统。你必须要熟悉DBR&#xff0c;下面我们就来看看文件系统解析DBR数据。 Dos Boot Record(DBR)操作系统引导记录是由操作系统的格式化程序建立的。在文件系统驱动操作不论什么一个磁盘卷时&#xff0c;这一部分的信息将被读取并作为文件系统在这个磁盘卷上的參数被使…

简单冒泡排序

将5个数字按从小到大排序。 #include <stdio.h> #include <stdlib.h> #include <math.h> int main() {int x[5] {0},temp 0;for(int i 0;i<5;i){scanf("%d",&x[i]);}//冒泡排序&#xff08;升序&#xff09;for(int j 0;j<4;j)//n个…

YouTube Java API入门

在本教程中&#xff0c;我将介绍Google的YouTube API &#xff0c;该API可让您使用YouTube的功能来启用应用程序。 YouTube是“杀手级”互联网应用程序之一&#xff0c;其流量占互联网总流量的很大一部分。 在开始之前&#xff0c;请确保您已阅读《 API概述指南》 。 我们将主…

mysql在mac上的坑

默认端口3306&#xff1f; 正确答案&#xff1a;3307 转载于:https://www.cnblogs.com/dudream/p/5375551.html

ServletContext图解

servlet之间共享数据资源&#xff01; 转载于:https://www.cnblogs.com/felixzh/p/4615902.html

C语言怎么输出百分号%

规律&#xff1a;printf函数中&#xff0c;当出现多个%时&#xff0c;由左至右&#xff0c;每两个%结合输出一个% #include <stdio.h> #include <stdlib.h> #include <math.h> int main() {int c 52;printf("% \n %% \n %%% \n %%%% \n %%%%% \n %%%%…

入侵Jasper以获取JSP页面的对象模型

为了对我的JSP进行一些检查和统计分析&#xff0c;我需要一个包含在其中的元素的类似于DOM的层次模型。 但是&#xff0c;解析JSP页面并不是一件容易的事&#xff0c;最好留给它一个出色的工具-Tomcat&#xff0c;Jetty&#xff0c;GlassFish以及其他所有工具都可以使用Jasper …

Linux自动化安装cobbler

1介绍 1.1 PXE PXE技术与RPL技术不同之处为RPL是静态路由&#xff0c;PXE是动态路由。RPL是根据网卡上的ID号加上其他记录组成的一个Frame&#xff08;帧&#xff09;向服务器发出请求。而服务器中已有这个ID数据&#xff0c;匹配成功则进行远程启动。PXE则是根据服务器端收到的…

iOS9适配系列教程

https://github.com/ChenYilong/iOS9AdaptationTips 转载于:https://www.cnblogs.com/zsw-1993/p/4879118.html

C语言形参

形参和实参区别 形参出现在函数定义中&#xff0c;在整个函数体内都可以使用&#xff0c;离开该函数则不能使用。实参出现在主调函数中&#xff0c;进入被调函数后&#xff0c;实参变量也不能使用。 形参和实参的功能是作数据传送。发生函数调用时&#xff0c;主调函数把实参…

避免延迟的JPA集合

Hibernate&#xff08;实际上是JPA&#xff09;具有集合映射&#xff1a; OneToMany&#xff0c; ManyToMany&#xff0c; ElementCollection。 所有这些默认情况下都是惰性的。 这意味着集合是List或Set接口的特定实现&#xff0c;其中包含对持久会话的引用&#xff0c;并且只…