【kubernetes】【基础资源使用】kubernetes中的Deployment使用

1 Why need Deployment?

K8S中Pod是用户管理工作负载的基本单位,Pod通常通过Service进行暴露,因此,通常需要管理一组Pod,RC和RS主要就实现了一组Pod的管理工作,其中,RC和RS的区别在于,RS提供更加高级的选择器(RS支持基于集合的选择符)。

在实际生产环境中,应用在初次部署完成后,当测试正常才会对外提供服务,之后就都是升级操作,而升级过程中,意味着删除老的Pod,并创建新的Pod。对于RS而言,如果需要升级Pod的镜像,可以使用kubectl edit修改RS中的Pod的镜像,或者直接修改RS的资源文件,然后执行kubectl apply,但是,这样做的话,会使得服务在短期内的不可用。虽然Service可以保证请求只路由到ready的Pod,并且RS会按照一定的策略删除老的Pod,然后再创建新的Pod,但是,在删除和创建的过程中可能无法保证服务的正常:无法保证永远有Pod在ready状态;无法保证有一定数量的Pod在处理请求。

为了解决服务更新的问题,K8S在RS的基础上提供了Deployment,它最大的特点是提供了滚动升级的能力,并且可以控制升级的速率,使得升级过程中可以正常提供服务,因此,通常都是使用Deployment,而不是RC或者RS。

2 Problem of ReplicaSet

设想一下,在只有RS的情况下,如何进行滚动升级,当然,在设计滚动升级流程之前必须要明确几个目标:

  • 升级过程中有固定数量的Pod提供服务
  • 为了保证服务不发生抖动,给定灰度的时间,可以减缓升级的速率
  • 如果升级过程中发现请求处理异常可以及时回退

为了实现这些目标,在只有RS的情况下,你可能会这么做(假设RS的replicas为3):

  • 用新的Pod的template创建新的RS,然后在RS的selector中新增标签,假设为A,并且RS的replicas设置为0
  • 在原来的RS的selector和Pod中都加一个标签,假设为B,这样的话,这些Pod就只被老的RS管理,并通过Service对外暴露
  • 将老的RS的replicas减1,新的RS的replicas加1,现在,old_replicas=2,new_replicas=1,并且由于A、B的存在,新老RS只管理各自的Pod,互不干扰,而这些Pod都可以被Service选择
  • 此时,Service的Pod同时有新老两个版本,这时候就可以通过监控查看成功率,如果没有问题,等待一段时间(相当于灰度时间)
  • 确认新的Pod没有问题,继续升级:老的RS的replicas继续减1,新的RS的replicas继续加1,现在old_replicas=1,new_replicas=2,继续关注监控指标
  • 继续升级:old_replicas=0,new_replicas=3,此时,所有的Pod都升级到了新版本

升级过程中总是有3个Pod提供服务,如果升级过程发现异常,可以直接修改old_replicas进行回退,同时,为了持续观察新的Pod的情况,在新的Pod创建完成后,不是立刻就对外提供服务,而是会等待一段时间,用于观察请求成功率,从而决定是否继续进行升级。

上面的过程比较复杂,而且灰度期间如果有问题,需要人工回滚,为了解决滚动更新的问题,K8S提供了Deployment,相比于ReplicaSet,它最大的优势就是滚动更新。

3 What is Deployment

从基础字段上看Deployment跟ReplicaSet没有任何区别,都是在Pod的基础上增加replicas字段,保证副本的数量,Deployment的主要功能也是保证副本的数量。但是,为了保证服务的可用性,相比于ReplicaSet,Deployment提供了滚动更新的能力。

Deployment提供的滚动更新跟上面的人工更新过程类似,也是创建额外的RS,然后修改新老RS的replicas值,然后再添加一些策略,例如,是全部一起更新,还是一个接一个的更新。

4 How to use Deployment

由于Deployment要创建多个Pod的副本,因此,Deployment中十分重要的字段就是PodTemplate和副本数:deployment.spec.template、deployment.spec.replicas。

剩下的字段基本都跟升级相关:

  • deployment.spec.progressDeadlineSeconds:当Deployment在一段时间内还没有升级成功,则会在deployment.status.conditions中的Progressing设置为False
  • deployment.spec.revisionHistoryLimit:保存用于回滚的历史版本数量,默认是10
  • deployment.spec.strategy:升级策略,这里有两种策略:Recreate、RollingUpdate。当deployment.spec.strategy.typeRollingUpdate时,表示滚动更新,并且可以设置deployment.spec.strategy.rollingUpdate.maxSurge(最大峰值,升级过程中可以创建出的超过预期pod数量的pod个数)和deployment.spec.strategy.rollingUpdate.maxUnavailable(最大不可用,升级过程中不可用的pod的数量,可以使用数字和百分比)。当deployment.spec.strategy.typeRecreate时,会干掉所有pod,当所有pod被干掉后,再创建新的pod。
kubectl create deployment --image=nginx my-nginx

上述语句创建了一个nginx的deployment,副本数和升级策略都是用的默认值:replcas=1,maxSurge=25%,maxUnavailable=25%。

4.1 Deployment相关资源和字段解释
kubectl get deploy my-nginx

上述命令可与查看该deployment,输出的结果中有三个重要字段:

  • READY:就绪的pod数量,
  • UP-TO-DATE:pod的定义与deployment中的Pod模板相同的pod的个数
  • AVAILABLE:可用的pod数量
4.2 Deployment的升级

Deployment通常通过更新镜像进行升级:

kubectl set image deploy my-nginx nginx=nginx:1.9.1
kubectl annotate deploy my-nginx kubernetes.io/change-cause="set nginx to 1.9.1"

上述语句先更新my-nginx这个Deployment的nginx容器的镜像,再使用annotate设置变更的原因,最后可以使用kubectl rollout status deploy my-nginx查看更新状态。

执行kubectl describe deploy my-nginx可以查看更新过程:

请添加图片描述

同时可以查看历史版本:

请添加图片描述

这里的CHANGE-CAUSE就是上面设置的annotate,另外,这里需要注意的是,这里只会保存不重复的版本,如果用两个镜像交替更新,这里只会保存两个版本。

4.3 Deployment的回滚

当升级过程中出现问题,如果是升级过程就卡主了,例如,pod没起来,或者拉取镜像失败,老的pod不会被干掉,此时也不会有啥问题。如果是程序的bug造成的问题,通过监控或者其他方式验证发现了问题,可以对Deployment进行回滚。

kubectl rollout undo deploy my-nginx

此时,deployment会回到上个版本,并且在rollout history中可以看到之前的版本换到了最新版本,版本号加1:

请添加图片描述

当然,也可以回滚到指定版本:

kubectl rollout undo deploy my-nginx --to-revision=3
4.4 Deployment的手动缩放

当pod数量不够或者过多时,可以手动修改Deployment的replicas配置:

kubectl scale deploy my-nginx --replicas=3

收到将Deployment的副本数设置为3。

5 小结

Deployment作为k8s中无状态部署的最常用的资源,所有的业务逻辑层的高可用和负载均衡都会用它来部署。当Deployment资源已经创建后,可以通过kubectl set imagekubectl scale更新镜像和副本数,还能够使用kubectl rollout undo进行回滚,同时,可以使用kubectl rollout history查看历史版本。

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

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

相关文章

【每日一题】528. 按权重随机选择

528. 按权重随机选择 - 力扣(LeetCode) 给你一个 下标从 0 开始 的正整数数组 w ,其中 w[i] 代表第 i 个下标的权重。 请你实现一个函数 pickIndex ,它可以 随机地 从范围 [0, w.length - 1] 内(含 0 和 w.length - 1&…

混合IT基础设施的安全挑战与缓解策略

自从“身份是新的边界”这句格言问世以来,公司已经开始扩展他们的能力和运营,超越了基于本地、办公室基础设施的范围。采用云原生技术意味着组织正在寻求扩大传统工作流程,而无需投入时间和资源来建立物理数据中心和其他硬件基础设施。 身份…

JTS:08 JTS图形相交

这里写目录标题 版本JTS disjoint intersects俩个图形不相交俩个图形 边相交俩个图形 内部相交俩个图形 点相交 版本 org.locationtech.jts:jts-core:1.19.0 链接: github JTS disjoint intersects 不相交的 九交模型FF*FF**** 相交的 九交模型 [T********] [*T*******] [**…

服务断路器_Resilience4j信号量隔离实现

POM引入依赖 <dependency><groupId>io.github.resilience4j</groupId><artifactId>resilience4j-bulkhead</artifactId><version>1.7.0</version> </dependency>信号量隔离修改YML文件 resilience4j:#信号量隔离bulkhead:ins…

GIT提示Another git process seems to be running in this repository

解决方法 1、进入项目里面的.git文件里面找到index.lock删除即可。

【算法】递归(高阶题目) -随时补充

文章目录 岛问题汉诺塔问题牛群繁衍数量问题求字符串的全部子序列字符串的全排列数字的全排列I数字的全排列IIN皇后IIN皇后I 岛问题 递归的方法: 遍历岛这个二维数组&#xff0c;如果当前数为1&#xff0c;则进入感染函数并将岛个数1感染函数&#xff1a;其实就是一个递归标注…

创建线程的4种方法

目录 一.前言 1.关于进程调度 (1)为什么要调度? (2)调度的真正对象 (3)调度的资源 2.线程 (1).线程的写法 (2)线程创建的方法 1.继承Thread (1)使用继承Thread,重写run的方式来创建线程 (2)继承Thread,使用匿名内部类 2.实现Runnable (1)使用实现Runnable,重写run…

自定义ElementPlus主题颜色

构建工具采用Vite CSS预处理器采用Sass 一.准备定制化的样式文件 1.安装Sass npm i sass -D 2.创建好文件目录 3.书写样式 ElementPlus默认样式. //index.scss/* 只需要重写你需要的即可 */ forward element-plus/theme-chalk/src/common/var.scss with ($colors: (prim…

pytorch固定随机数中种子

1、添加到yolov7的utils/general.py文件最下面 import pkg_resources as pkg def check_version(current0.0.0, minimum0.0.0, nameversion , pinnedFalse, hardFalse, verboseFalse):# Check version vs. required versioncurrent, minimum (pkg.parse_version(x) for x in …

【Verilog 教程】6.6Verilog 仿真激励

关键词&#xff1a;testbench&#xff0c;仿真&#xff0c;文件读写 Verilog 代码设计完成后&#xff0c;还需要进行重要的步骤&#xff0c;即逻辑功能仿真。仿真激励文件称之为 testbench&#xff0c;放在各设计模块的顶层&#xff0c;以便对模块进行系统性的例化调用进行仿真…

c#用Gnuplot画图源码

直接调用这个类即可&#xff0c;需要下载个GnuPlot安装下。 // Author: Leonardo Tazziniusing System; using System.Diagnostics; using System.Drawing; using System.IO; using System.Windows.Forms;/// <summary> /// Tested with Gnuplot 5.2 /// </summary&g…

云原生微服务治理经典框架之Spring Cloud Alibaba核心技术与实战案例

系列文章目录 送书第一期 《用户画像&#xff1a;平台构建与业务实践》 送书活动之抽奖工具的打造 《获取博客评论用户抽取幸运中奖者》 送书第二期 《Spring Cloud Alibaba核心技术与实战案例》 文章目录 系列文章目录1、云原生如何做微服务治理&#xff1f;2、微服务治理框…

类和对象:运算符重载

本篇文章来介绍一下C中的运算符重载&#xff0c;以及与运算符重载有关的三个默认默认成员函数&#xff1a;赋值运算符重载&#xff0c;普通对象取地址与const对象取地址操作符重载&#xff0c;也就是下面图片中6个默认成员函数的后三个&#xff0c;前三个默认成员函数在之前文章…

【go语言】结构体

结构体 结构体定义 type name struct{value1 type1value2 type2...... }组成结构体的数据称为字段&#xff0c;每一个字段有一个类型和一个名字&#xff0c;每一个字段的名字必须唯一&#xff0c;字段的类型任意。 创建结构体 type myStruct struct{i1 intf1 float32str st…

HDLBits-Edgedetect

刚开始写的代码如下&#xff1a; module top_module (input clk,input [7:0] in,output [7:0] pedge );reg [7:0] in_pre;always (posedge clk)begin in_pre < in;endassign pedge in & ~in_pre; endmodule但是提交结果是错误的。猜想原因如下&#xff1a; assign p…

某音网页端 X-Bogus 参数

逆向目标 目标&#xff1a;某音网页端用户信息接口 X-Bogus 参数 接口&#xff1a;aHR0cHM6Ly93d3cuZG91eWluLmNvbS9hd2VtZS92MS93ZWIvdXNlci9wcm9maWxlL290aGVyLw 什么是 JSVMP&#xff1f; JSVMP 全称 Virtual Machine based code Protection for JavaScript&#xff0c;即 …

【网络八股】TCP八股

网络八股 请简述TCP/IP模型中每层的作用&#xff0c;典型协议和典型设备介绍一下三次握手的过程介绍一下四次挥手的过程必须三次握手吗&#xff0c;两次不行吗&#xff1f;为什么ACK数据包消耗TCP的序号吗三次握手中可以携带应用层数据吗四次挥手时&#xff0c;可以携带应用层数…

crypto:丢失的MD5

题目 得到一个md5.py 运行一下&#xff0c;发现报错&#xff0c;修改一下 运行之后又报错 报错原因是算法之前编码 正确的代码为 import hashlib for i in range(32,127):for j in range(32,127):for k in range(32,127):mhashlib.md5()m.update((TASC chr(i) O3RJMV c…

【Java SE】Lambda表达式

目录 ♫什么是Lambda表达式 ♫Lambda表达式的语法 ♫函数式接口 ♫Lambda表达式的使用 ♫变量捕获 ♫ Lambda表达式在集合中的使用 ♪Collection的foreach()&#xff1a; ♪List的sort()&#xff1a; ♪Map的foreach() ♫什么是Lambda表达式 Lambda 表达式是 Java SE 8中一个…

SpringMVC 学习(二)Hello SpringMVC

3. Hello SpringMVC (1) 新建 maven 模块 springmvc-02-hellomvc (2) 确认依赖的导入 (3) 配置 web.xml <!--web/WEB-INF/web.xml--> <?xml version"1.0" encoding"UTF-8"?> <web-app xmlns"http://xmlns.jcp.org/xml/ns/javaee…