容器花絮:什么时候应该将应用程序切分为多个容器?

本文讲的是容器花絮:什么时候应该将应用程序切分为多个容器,【编者的话】本文讨论了如何将一个应用切分成为多个容器,有哪些需要思考的问题?又有哪些需要避免的思维误区?文中都有涉及。

围绕着应该将应用程序的哪些部分切分为多个容器以及为什么要这样做,存在着很多困惑。我最近对Docker用户邮箱列表的回应促成了今天的文章。在这篇文章中我打算评估一个镜像化的Java应用程序,它历史上曾经运行在一个单一的Tomcat服务器里,并解释为什么我会把它切分为独立的容器。为了让事情有趣——我也旨在证明这个动作(比如将应用程序分割为独立的容器)和数据及(工程)逻辑,而不是简单地说“有一个原则”,即必须在所有时间都遵守。

让我们看一个Java应用程序示例,由以下两部分组成:

  1. 基于Struts Web框架构建的一个前端应用
  2. 基于J2EE构建的一个后端REST API

如前所述,这个应用程序历史上曾经跑在一个Tomcat服务器里,两个组件是通过一个基于REST的API进行通信的,所以问题变成了:

我应该要将这个应用程序切分成多个容器吗?

答案是肯定的,我相信这个应用程序应该切分为两个不同的Docker容器,但这应该经过慎重考虑之后。

而不是“仅仅因为”将应用程序切分成多个容器或试图坚持一些新奇的原则(比如“每个容器只运行一个进程”)——我建议我们应该通盘考虑项目需求然后做出明智和聪明的决定。是否所有的应用程序都应该切分为多个容器?容器化通过提供一个更简单的部署策略至少会使你的软件生涯更便捷。

让我们在示例应用程序的分析上做些短暂的停留,做一些设计思考

  1. JVM是多线程的,所以没有必要运行多个Unix/Linux进程。事实上,我认为这是Java工程师理所当然的困惑。从历史上看,Java开发人员实际上喜欢在同一个JVM里边运行多个应用程序,在实践中,这样做可以节省相当多的内存。此外,Web应用服务器比如Tomcat是打从一开始就支持在单个JVM上运行多个应用程序。这实际上是运行一个简单的Java程序相对于Java EE应用程序(这可能是由多个线程和多个不同的程序组成)的主要区别。
  2. 在现实中,许多应用在每个容器中使用多个进程,Apache Web服务器的PreforkMPM和模块都在同一个容器中使用多个进程。现代的Web应用具有事件驱动编程反应器模式的特性(例如Nginx),实际衍生了许多子进程。我认为,FastCGIAIO以及反应器模式的整个理念是将工作分担给其他进程(或线程),而让内核专注处理I/O。Linux内核是相当擅长子进程调度的,Kubernetes、Swarm和单个的Docker容器不太擅长干这活。进程(及线程)都是有关内核资源分配的,而容器涉及到了集群资源分配。
  3. 运行单容器单进程的“最佳实践”被广泛认为是“原则“,但听起来更多像哲学。作为一名工程师,我想要理解技术组件,并做出合乎逻辑的决定。我认为,这种”最佳实践“甚至不是普遍的共识,其控制应用程序根源于广泛缺乏了解的Unix是如何工作的。
  4. Linux容器在历史上有许多种形式,许多实际推荐运行的是在单个容器里运行多个进程。什么使得Docker容器与众不同的呢?Linux容器本质上就是对于系统调用的克隆、SElinux和Cgroups,它们是否是LXC或者Docker(通过libcontainer)类型是无关紧要的,因为Linux内核自身就可以处理进程隔离。
  5. 在有效的时间之内通过Socket、文件或者网络等处理通信,每一个方法都有它各自的优点和缺点。给定应用程序的通信方式肯定会影响你是否想将您的应用程序切分成多个容器。
  6. 代码、配置和数据的分离也会影响您的应用程序切分为多个容器的能力。如果你的应用程序具有良好的代码、配置和数据的分离能力,将会非常容易切分为多个容器,如果你的应用代码非常古老,并且不太好理解,将会对于文件系统带来不必要的改变,并且非常难以作切分,注意,下面的方法是非常棒的!你不必重写你的应用来容器化,而可以通过将你的应用放入一个容器就可以享受Docker容器格式的好处。你可以轻松地移动(使用注册中心服务)和部署(执行docker run命令)。

好的,有一些Unix应用遇到101错误了,让我们回到示例Java应用的分析:

  1. 如上所述的两个Java组件看起来做的都是不同的事情,一个组件是Web前端,另外一个是API服务器。既然这些组件做的是不同的事情(比如他们确实是不同的服务),几乎没有机会比在相同的JVM里有性能优势(当然了,没有实际的测试性能,我不能100%确定)。
  2. 这两个应用程序使用REST API来进行通信(而不是使用Socket、共享内存或文件等方式)。
  3. 一般来说,如果一个应用程序包含一个API层和一个前端层,对于独立地扩展这些应用就非常有用。比如说如果一个API也被一个移动应用所消费,根据用户负载进行动态伸缩就很有用,而基于Web的前端就不需要伸缩。相反,如果我扩展web前端,我可能还需要扩展API服务器的部分,但每五个Web前端我可能只需要再多一个API服务器。长话短说,伸缩逻辑复杂,借助于类似Kubernetes这样的工具进行独立的伸缩是非常有用的。

基于这三个单独的观察,我推荐将这两个组件切分为单独的容器,我还推荐使用容器编排工具比如Kubernets或者OpenShift来将这些服务连接在一起。我不会将这一决定基于所谓的“原则”或“最佳实践”,而是会基于它的应用架构和某种形式的信息推理。

这儿的事情变得疯狂,我提交(给你)一个新的“最佳实践”,请击鼓: 是的,如果您的应用程序/服务具有很好的代码、配置和数据的隔离,安装非常清晰(因为安装脚本会使整个过程变得困难),提供了一个非常清晰的通信模式,就非常有意义来切分/分配为单容器运行单个服务。

从根本上说,我建议我们所有人都应该开始更加理性地考虑如何把应用放入容器里,意识到容器化不仅仅是一个哲学理论,更是可以解决技术痛点问题的。我喜欢容器,并且在所有的时间都愿意使用它们,但是也是以一个睿智的方式。我确信人们对于容器化应用的意见和想法,我鼓励您在下面的评论栏中分享你的想法。

原文链接:Container Tidbits: When Should I Break My Application into Multiple Containers?(翻译:胡震)

原文发布时间为: 2016-04-11

本文作者:国会山上的猫TuxHu 

本文来自合作伙伴DockerOne,了解相关信息可以关注DockerOne。

原文标题:容器花絮:什么时候应该将应用程序切分为多个容器?

时间: 2024-08-08 02:02:00

容器花絮:什么时候应该将应用程序切分为多个容器?的相关文章

javase-关于马士兵老师的坦克大战小程序为什么用了LIST容器来装炮弹还是只能一发一发的打

问题描述 关于马士兵老师的坦克大战小程序为什么用了LIST容器来装炮弹还是只能一发一发的打 主窗口TankClient.java代码如下:import java.awt.*;import java.awt.event.*;import java.util.List;import java.util.ArrayList; public class TankClient { public static void main(String[] args) { new MyFrame(""Tan

纯CSS实现下拉菜单及下拉容器等(纯CSS实现导航条及导航下拉容器)

原文:纯CSS实现下拉菜单及下拉容器等(纯CSS实现导航条及导航下拉容器)  虽然网上类似甚至相同的案例有很多,但是我还是写下,以记下笔记,也可供大家参考 希望大家可以指导批评~~ 首先我们以列表ul li 来开始我们菜单也可以说导航条的制作: 在页面中我们首先构建以下XHTML结构: <body> <ul id="navWrapper"> <li> <a href="#">Menu A</a> <u

容器到底可以做什么?—客户不再头疼 —— OSCAR即将发布“容器技术应用场景”规范

容器产业化的三个问题 现在的容器真是让人欢喜让人忧,在客户那里,厂商经常表示容器无所不能,但又有很多客户吐槽容器体验不好.所以,容器你想闹哪样? 容器表示很无辜,我本将心向明月,奈何明月"掉"沟渠.好好的明月技术,用不好,真是有很多的Gap. 不可否认,容器现在还处在初级发展阶段,仍有很多需要完善的地方,但是目前市场上也有很多成功使用容器的案例. 容器能做什么? 怎么做? 怎么选技术解决方案? 这是客户需要亟需了解的三个问题. 容器技术能够帮助企业解决什么具体问题?企业在部署容器技术时

构建插件式的应用程序框架(四)-服务容器

IApplication接口是派生于IServiceContainer接口的.为什么要派生于IServiceContainer呢?我们来看看IServiceContainer的定义,它有几个AddService方法和RemoveService方法以及从IserviceProvider继承过来的GetService方法.Service本身是.NET设计时架构的基础,Service提供设计时对象访问某项功能的方法实现,说起来还真拗口.就我看来,ServiceContainer机制的本质就是解耦合,就

容器如何访问外部世界?- 每天5分钟玩转 Docker 容器技术

下面我们讨论容器如何与外部世界通信.这里涉及两个方向: 容器访问外部世界 外部世界访问容器 容器访问外部世界 在我们当前的实验环境下,docker host 是可以访问外网的. 我们看一下容器是否也能访问外网呢? 可见,容器默认就能访问外网. 请注意:这里外网指的是容器网络以外的网络环境,并非特指 internet. 现象很简单,但更重要的:我们应该理解现象下的本质. 在上面的例子中,busybox 位于 docker0 这个私有 bridge 网络中(172.17.0.0/16),当 busy

.NET程序在Linux容器中的演变

本文讲的是.NET程序在Linux容器中的演变[编者的话]Linux容器技术已被开发人员所熟知,现在.NET程序可以跑在Docker容器中,这为以Windows中心的开发人员带来了好处. [上海站|3天烧脑式微服务架构训练营]培训内容包括:DevOps.微服务.Spring Cloud.Eureka.Ribbon.Feign.Hystrix.Zuul.Spring Cloud Config.Spring Cloud Sleuth等. 本文将首先讨论镜像的构建时间和启动时间,接着会将一个简单的.N

在 Docker 容器中运行应用程序

案例说明 运行 3 个容器,实现对网站的监控. 三个容器的说明: 容器 web: 创建自 nginx 映像,使用 80 端口,运行于后台,实现 web 服务. 容器 mailer: 该容器中运行一个 mailer 程序,运行于后台,当接收到事件后会向管理员发送邮件. 容器 agent: 该容器运行一个 watcher 程序,以交互模式运行,用于不断地监测 web 服务的运行情况,一旦出现故障会立即向 mailer 容器发送消息. 创建容器 创建并运行 web 容器 $ docker run --

java容器 collection-下面这个小程序打印出的为什么是true和false啊,求大神分析一下,万分感谢~

问题描述 下面这个小程序打印出的为什么是true和false啊,求大神分析一下,万分感谢~ public class BasicContainer { public static void main(String[] args) { Collection c = new HashSet(); c.add("hello"); c.add(new Name("lu","dongdong")); c.add(new Integer(100)); Sys

IT风向标:应用程序容器正入侵数据中心

对于基础设施和运营团队而言,应用程序容器的发展是一把双刃剑. 哪儿来的这么大动静?其实是由开发人员研发的数以百万计的Docker应用程序容器产生的,并以迅雷不及掩耳之势直侵企业数据中心. 的确,开发人员利用Docker的热情从未如此之高.应用程序容器项目和公司在2013年出炉,但它的价值已经超过10亿美元.Docker容器已经被下载超过4亿次,在Docker 中心已经有超过100000个"Dockerized"应用程序.Docker应用程序容器产生的,并以迅雷不及掩耳之势直侵企业数据中