浅谈如何打造对容器友好的应用

本文原载于 https://blog.shifudao.com, 原文链接

前言

容器化是一种全新的产品交付方式,几乎是颠覆了传统的软件交付方式和交付流程,并且在各大公司都已经有了成熟的应用,自从k8s问世之后,容器化的交付方式得到了前所未有的高速发展。k8s的资源调度有效提高了资源率利用率,充分利用每台Node节点的计算能力,大大降低了运维压力,以及降低了基础设施的总体成本。

本文就来谈谈如何打造对容器友好的应用,以便现有的应用可以很容易逐步迁移到容器中,实现容器化的交付,方便未来逐步向k8s迁移。

容器运行方式

简述一下Docker容器的运行方式,Docker容器会在系统中运行一个隔离环境,你需要打包所有运行依赖到容器镜像中。同时,Docker允许向容器内注入运行时需要的环境变量,以便修改应用程序的运行行为而无需重新打包镜像。特别注意Docker强调容器镜像应该尽可能轻量级,最佳实践要求一个容器只运行一个进程,并且保持该进程在容器内的PID为1

一图说明虚拟机和Docker容器的对比:

图片来源: https://www.nakivo.com/blog/docker-vs-kubernetes/

如何让应用程序对容器友好

容器可以帮忙解决程序的运行依赖问题,但同时应用程序本身也必须对容器环境进行适配,否则盲目迁移到容器反而会造成更大的迁移成本,甚至无法适配容器环境导致运行失败。本文就浅谈下如何适配容器环境。

依赖轻量化

容器需要包含整个程序的运行依赖,一个打包好的镜像需要在全网节点分发部署。为了减小容器的容量,加快部署速度,有必要对应用程序本身的依赖进行精简。以下都是比较有效且合理的手段:

  • 微服务化
  • 前后端分离
  • 打包镜像中尽可能保证每一层容量最小 (参考官方最佳实践)

除了最后一项需要编写Dockerfile的人对Docker本身有比较好的理解之外,前两项都是应用程序设计本身需要解决的问题。

独立运行

Docker容器的最佳实践要求每个容器只运行一个进程,该进程pid必须为1(比如通过exec就能保证java进程的pid为1 CMD exec java ${JAVA_OPTS} -jar xxx.jar),所以要求每个程序都能独立运行,而不是需要一个辅助进程在旁协助。

这就要求应用程序自身就能对运行环境做一些自检和适配,而不是需要后台跑个脚本帮助应用程序稳定运行(尽管很多早期的应用通常这么干)

支持环境变量注入

在容器中运行的应用最简单修改配置的方式就是注入环境变量,因此要求应用程序必须能支持环境变量。这一点很多现代化的开发框架都已经支持,比如Spring Boot的配置就已经支持通过环境变量直接覆盖application.yml定义好的配置,参考官方文档,另外像Logback这一类的日志框架,也支持通过环境变量覆盖诸如日志级别之类的可配置项,如果已有的技术栈支持环境变量覆盖,直接复用即可,简单做一些相应的优化配置即可完成

状态分离

容器对于 无状态 应用非常友好,而对于 有状态 的应用维护起来要麻烦的多。如果你的应用程序有状态,那么需要考虑状态分离,将无状态的部分和有状态的部分剥离成不同的应用,分别部署。无状态的应用可以部署到容器集群中,有状态的应用考虑单独部署一个独立的集群维护。

例如应用程序需要支持存储,那么可以考虑将存储单独剥离成一个存储服务(例如使用现有的云存储服务,或者使用数据库存储等等);如果应用程序需要支持session,可以考虑改造为无sessionJWT等等。

这样去状态化之后,就可以使用容器部署无状态的应用了。

处理多时区问题

在本地开发很少有人会关心时区问题,很多人都习惯了本地时区,但是迁移到容器环境中非常容易忽略时区问题,造成时间差。

由于容器环境默认使用UTC时间,和本地时间相差8小时,所以应用程序必须对时区进行处理。这通常包含两方面的内容:

  • 应用程序本身需要处理时间日期的部分
  • 数据库的时间日期存储

数据库的最佳实践也是要求存储带时区的日期格式,但是大部分开发人员容易忽略,图省事存储为不带时区的格式。当应用程序和数据库所在时区不同就会造成问题。

当然为了简单起见,应用程序也无需考虑国际化的时候,有2种办法可以不用编码规避这个问题:

  • 在构建Docker镜像的时候修改镜像的时区,参考这篇问答
  • 注入特殊环境变量或properties(仅对特定程序有用)

对于第一种方案,会额外增加一个Layer,并且对于不同的base image修改时区的方法也不一样(如Debian和Alpine修改时区的方法就不一样)。

对于第二种方案,需要考虑具体的应用程序,比如JVM支持注入user.timezone这个system property修改时区:

java -Duser.timezone=America/New_York

对于Nodejs则支持通过TZ环境变量指定时区:

TZ='Europe/Amsterdam' node server.js

尾声

本篇文章浅谈了如何打造对容器友好的应用需要注意的一些地方。只是想将应用迁移到容器运行的话还是比较容易的,差不多以上几点注意到之后就可以简单在容器环境启动一个单实例容器镜像跑起来了。如果想做到更加适配容器集群环境的话,通常还需要考虑到以下一些问题:

  • 日志打印到终端,而不是存储日志文件
  • 多实例如何协调一致。如接口幂等,防止用户请求分配到不同实例造成数据处理异常,多实例负载均衡,而不是互相冲突等等
  • 容器探针。需要应用程序自己实现一个活性探测接口(如http, tcp, 脚本探测等)
  • ……

容器的出现带来了一种全新的交付方式,k8s的出现将容器的热度大幅度拉升,基于容器和k8s的交付方案层出不穷。为了赶上这波热度,我本人也做了很多实际的体验与调研,特此总结此篇文章,以供其他对容器化有兴趣的技术人员参阅。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容