Java程序员如何快速理解Kubernetes


我们希望微服务是可复制的,可替换的工作节点,这样可以轻松进行升级或降级,同时无需任何停机时间,并花费最少代价的管理。我们可以说我们希望他们成为我们的小黄人(minions)。本文我们将通过一个简单的例子来了解Kubernetes可以通过创建和编排一群“小黄人"来为我们做些什么。您可以与本文一起编码或从此处[1]克隆项目。
先决条件


需要将使用Docker容器化微服务以便在Kubernetes中运行它们。我们将使用Minikube,而不是使用云托管的Kubernetes,以便可以在本地沙箱运行。
目的


我们的小黄人军团将是Java微服务。我们希望军团中有不同类型的工作角色,以便能够了解Kubernetes可以为我们做些什么。因此,我们的目标是让每个微服务都响应一个简单的http请求,其响应如下:

使用ASCII字来表示minion的类型。
构建Java Minion服务


我们可以通过Spring Boot Web应用程序来启动我们的微服务,程序使用具有Web启动依赖性的Spring Initializr初始化:

在项目中,创建一个使用@RestController注释的Controller来处理请求。使用@RequestMapping(method = GET)来提供响应主体。所以首先我们可以这样做:
@RequestMapping( method=GET)

@ResponseBody

public String minion() throws UnknownHostException {

   StringBuilder stringBuilder = new StringBuilder();

   stringBuilder.append("Host: ").append(InetAddress.getLocalHost().getHostName()).append("<br/>");

   return stringBuilder.toString();

}

但这并不能完全满足需求。我们可以输出ASCII字,但选择哪种minion类型?为此可以使用一个技巧。创建一个可以采用我们选择的任何minion类型的应用程序。要做到这一点,需要它包含一个ASCII艺术字库。因此,我们创建了一个名为MinionsLibrary的类,使用@Component注解,在内部我们创建了一个地图,我们使用此博客[2]中的一些minions初始化:
@Component

public class MinionsLibrary {

    private Map<String,String> map = new HashMap<>();

    public MinionsLibrary(){

      map.put("one-eyed-minion",<COPY-PASTE MINION ASCII ART HERE>);

      map.put("two-eyed-minion",<COPY-PASTE MINION ASCII ART HERE>);

      map.put("sad-minion",<COPY-PASTE MINION ASCII ART HERE>);

      map.put("happy-minion",<COPY-PASTE MINION ASCII ART HERE>);

    }

}

或者你可以从https://github.com/ryandawsonuk/minions/tree/master/src/main/java/org/minions/demo获取。
然后告诉微服务是哪种minion类型。使用Spring应用程序的名称属性(我们稍后可以使用Docker环境变量设置)来执行此操作。它还将帮助我们稍后在响应中显示我们的应用程序版本,所以现在的Controller变为:
@RestController

public class Controller {

    private final String version = "0.1";

    private MinionsLibrary minionsLibrary;

    @Value("${spring.application.name}")

    private String appName;

    public Controller(MinionsLibrary minionsLibrary){

        this.minionsLibrary=minionsLibrary;

    }

    @RequestMapping( method=GET)

    @ResponseBody

    public String minion() throws UnknownHostException {

        StringBuilder stringBuilder = new StringBuilder();

        stringBuilder.append("Host: ").append(InetAddress.getLocalHost().getHostName()).append("<br/>");

        stringBuilder.append("Minion Type: ").append(appName).append("<br/>");

        stringBuilder.append("IP: ").append(InetAddress.getLocalHost().getHostAddress()).append("<br/>");

        stringBuilder.append("Version: ").append(version).append("<br/>");

        stringBuilder.append(minionsLibrary.getMinion(appName));

        return stringBuilder.toString();

    }

}

现在选择'image'包以匹配应用程序名称,该名称将是minion类型名称(例如'单眼小黄人')。
容器化并部署


需要为我们的应用程序创建一个Docker镜像。我们想在Docker镜像中构建可执行的jar,然后在容器启动时启动Java应用程序。可以使用多阶段Docker构建来完成此任务。 Dockerfile是:
FROM maven:3.5-jdk-8 as BUILDMINION

COPY src /usr/src/myapp/src

COPY pom.xml /usr/src/myapp

RUN mvn -f /usr/src/myapp/pom.xml clean package -DskipTests

FROM openjdk:alpine

COPY --from=BUILDMINION /usr/src/myapp/target/*.jar /maven/

CMD java $JAVA_OPTS -jar maven/*.jar

从开始到'FROM openjdk:alpine'是构建JAR,然后jar包被拷贝到基于轻量的openjdk:alpine镜像的下一阶段构建。使用JAVA_OPTS参数来限制程序的内存占用(关于降低内存,可以参考该文章[3])。
然后使用命令“docker build . -t minion”构建一个镜像。
通过创建Kubernetes部署文件来部署它。我们称之为“minion-army.yml”。这将包含每个minion类型的条目。这是其中的一个minion类型:
apiVersion: apps/v1beta1

kind: Deployment

metadata:

 name: one-eyed-minion

labels:

   serviceType: one-eyed-minion

spec:

 replicas: 2

template:

   metadata:

     name: one-eyed-minion

     labels:

       serviceType: one-eyed-minion

   spec:

     containers:

       - name: one-eyed-minion

         image: minion:latest

         imagePullPolicy: Never

         ports:

         - containerPort: 8080

         env:

         - name: JAVA_OPTS

           value: -Xmx64m -Xms64m

         - name: SPRING_APPLICATION_NAME

           value: "one-eyed-minion"

---

apiVersion: v1

kind: Service

metadata:

 name: one-eyed-minion-entrypoint

namespace: default

spec:

 selector:

   serviceType: one-eyed-minion

ports:
   - port: 8080

     targetPort: 8080

     nodePort: 30080

type: NodePort

请注意,“SPRING_APPLICATION_NAME”变量会自动与spring.application.name属性匹配,以便此minion服务成为单眼小黄人类型。有两个这种minion类型的实例(副本)可用,Kubernetes服务将自动将请求路由到其中一个或另一个。
该服务将通过Minikube以端口30080暴露对外提供服务 (对于真正的Kubernetes,该服务的这一点会有所不同,因为我们使用LoadBalancer而不是NodePort,并且不会限制在minikube端口范围)。服务将使用与服务匹配的Pod来处理它。我们将为每种类型提供一种服务。
minion类型的部署将创建两个Pod。每个人都是这种类型的工作节点。
我们可以为每个minion类型重复上面的配置,每次增加外部端口号以便使用不同的端口,或者我们可以使用这个GitHub存储库,它还具有其他配置,可以在不停机的情况下进行小型版本升级(如果我们使用Helm,我们可以避免重复,但我们不想添加比我们更多的工具)。
创建军团


首先启动mMinikube:
minikube start --memory 4000 --cpus 3

等待它开始,然后将您的Docker registry链接到Minikube,并为Minikube构建minion图像:
eval $(minikube docker-env)

docker build . -t minion

然后我们可以部署军团:
kubectl create -f minion-army.yml

并看到类型:
open http://$(minikube ip):30080

open http://$(minikube ip):30081

open http://$(minikube ip):30082

open http://$(minikube ip):30083

每个看起来都很像文章开头的快乐小黄人页面。
我们可以通过“kubectl get pods”来查看整个军队,或者“minikube dashboard”进到Pods页面:

创造更多的部队


我们可以在minikube dashboard的Deployments部分下创建更多特定类型的minions:

一个小黄人倒下,另一个替补他的位置


假设从浏览器点击快乐小黄人服务时得到的:

如果杀死“happy-minion-58c9c46d67-j84s9”会发生什么?可以通过仪表板的Pod部分删除:
kubectl delete pod happy-minion-58c9c46d67-j84s9

如果你在浏览器中点击刷新几次(杀死小黄人兵可能需要一点时间),你会看到该服务会使用该类型的另一个小黄人。如果浏览Pod部分,您将看到Kubernetes创建了一个新的Pod来代替您删除的那个,以保证该部署中有两个节点。
Minion升级


我们还可以为小黄人进行滚动升级。为此,我们应该在minions-army.yml文件的每个Deployment部分的'spec'部分下面(它可以直接位于同一级别的'replicas'下面):
minReadySeconds: 10

strategy:

   type: RollingUpdate

   rollingUpdate:

     maxUnavailable: 1

     maxSurge: 1
然后将Controller类中的版本更改为0.2,保存它然后执行:
docker build . -t minion:0.2

然后打开minion-army.yml并找到 - 用“0.2”替换所有“最新”,保存更改并执行:
kubectl apply -f minion-army.yml --record

刷新其中一个minion类型的浏览器,以查看版本更改是否与kubectl rollout status部署中看到的内容一致,其中是minion类型(例如one-eyed-minion)。
小黄人回滚


要查看已部署的历史记录,请执行kubectl rollout history deployment ,回滚执行 kubectl rollout undo deployment --to-revision = 1(可能需要一段时间)。
销毁军团


用以下方法摧毁军队:
kubectl delete -f minion-army.yml

用“minikube stop”停止minikube。
相关链接:
https://github.com/ryandawsonuk/minions/blob/master/minion-army.yml

http://textart4u.blogspot.co.uk/2013/08/minions-emoticons-text-art-for-facebook.html

https://dzone.com/articles/how-to-decrease-jvm-memory-consumption-in-docker-u
 

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

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

相关文章

NLP基础 : HMM 隐马尔可夫模型

Hidden Markov Model, HMM 隐马尔可夫模型&#xff0c;是一种描述隐性变量(状态)和显性变量(观测状态)之间关系的模型。该模型遵循两个假设&#xff0c;隐性状态i只取决于前一个隐性状态i-1&#xff0c;而与其他先前的隐形状态无关。观测状态也只取决于当前的隐形状态。因此我们…

关于秒杀系统优化方向

今天听了一节咕泡学院的公开课&#xff0c;有收获。 秒杀系统的特点&#xff1a; 1.限时&#xff1b;2.限量供应&#xff1b;3.并发量大&#xff1b;如何优化&#xff1a; 1.客户端数据缓存。 2.CDN加速。 3.nginx动静分离&#xff0c;静态资源缓存&#xff0c;负载均衡。 4.se…

Mysql插入很慢,找到了稍微快点的方法

MYSQL批量插入数据库实现语句性能分析 假定我们的表结构如下 代码如下 CREATE TABLE example ( example_id INT NOT NULL, name VARCHAR( 50 ) NOT NULL, value VARCHAR( 50 ) NOT NULL, other_value VARCHAR( 50 ) NOT NULL ) 通常情况下单条插入的sql语句我们会这么写&…

Linux - 时间相关命令 - ntpdate, date, hwclock

1. 概述 最近也不知道写啥了, 把之前的老文档整理一下, 凑个数什么的配置时间这种工作, 偶尔还是要用一下主要描述 3 个命令的简单适用 ntpdatehwlock2. ntpdate 1. 概述 用于同步时钟的命令2. 机制 通常是有一个服务器对外提供时间客户端可以与时间服务器同步ntp 是他们之间交…

RUNOOB python练习题1

用来练手的python 练习题&#xff0c;原链接 : python练习实例1 题干 : 有四个数字&#xff1a;1、2、3、4&#xff0c;能组成多少个互不相同且无重复数字的三位数&#xff1f;各是多少&#xff1f; import numpy as np cen np.array([1,2,3,4]) tens np.array([1,2,3,4])…

mysql explain用法和结果的含义

explain显示了mysql如何使用索引来处理select语句以及连接表。可以帮助选择更好的索引和写出更优化的查询语句。 使用方法&#xff0c;在select语句前加上explain就可以了&#xff1a; 如&#xff1a; explain select surname,first_name form a,b where a.idb.id EXPLAIN列…

日志模块logging用法

一、常用日志记录场景及最佳解决方案&#xff1a; 日志记录方式 最佳记录日志方案 普通情况下&#xff0c;在控制台显示输出 print() 报告正常程序操作过程中发生的事件 logging.info()(或者更详细的logging.debug()) 发出有关特定事件的警告 warnings.warn()或者loggin…

MySQL 亿级数据需求的优化思路(一),交易流水记录的查询

对MySQL的性能和亿级数据的处理方法思考&#xff0c;以及分库分表到底该如何做&#xff0c;在什么场景比较合适&#xff1f; 比如银行交易流水记录的查询 限盐少许&#xff0c;上实际实验过程&#xff0c;以下是在实验的过程中做一些操作&#xff0c;以及踩过的一些坑&#…

RUNOOB python练习题2

用来练手的python 练习题&#xff0c;原链接 : python练习实例2 题干 : 企业发放的奖金根据利润提成。利润(I)低于或等于10万元时&#xff0c;奖金可提10%&#xff1b;利润高于10万元&#xff0c;低于20万元时&#xff0c;低于10万元的部分按10%提成&#xff0c;高于10万元的…

dubbo负载均衡策略和集群容错策略

dubbo负载均衡策略 random loadbalance 默认情况下&#xff0c;dubbo是random load balance随机调用实现负载均衡&#xff0c;可以对provider不同实例设置不同的权重&#xff0c;会按照权重来负载均衡&#xff0c;权重越大分配流量越高&#xff0c;一般就用这个默认的就可以了。…

MySQL 亿级数据需求的优化思路(二),100亿数据,1万字段属性的秒级检索

最近在研究亿级数据的时候&#xff0c;无意中看到了一个关于写58同城的文章 https://blog.csdn.net/admin1973/article/details/55251499?fromtimeline 其实上面讲的versionext的方式以及压缩json的思路&#xff0c;对于我来讲都可以看得懂&#xff0c;想得通&#xff0c;其…

RUNOOB python练习题3

用来练手的python 练习题&#xff0c;原链接 : python练习实例3 拿到题目就写了如下代码&#xff0c;思路是因为使用**0.5进行开平方操作时&#xff0c;python会将数据类型自动转换为float单精度浮点型。这里利用提取其整数部分&#xff0c;来判断这个数是否是完全平方数。 z…

使用git将项目上传到github(最简单方法)

使用git将项目上传到github&#xff08;最简单方法&#xff09; 首先你需要一个github账号&#xff0c;所有还没有的话先去注册吧&#xff01; https://github.com/ 我们使用git需要先安装git工具&#xff0c;这里给出下载地址&#xff0c;下载后一路直接安装即可&#xff1…

数据库 概念详解

数据库 概念详解 一、MySQL MySQL 事务 MySQL 锁 MySQL 二、Redis 三、MongoDB 四、Memcached 转载于:https://www.cnblogs.com/guozepingboke/p/10743648.html

RUNOOB python练习题4

用来练手的python习题其四&#xff0c; 原题链接: python练习实例4 题干: 输入某年某月某日&#xff0c;判断这一天是这一年的第几天&#xff1f; 这个题目比较简单&#xff0c;只需要注意闰年和非闰年的区别就可以了。我这里使用numpy矩阵存储每个月的天数&#xff0c;之后用…

GitHub入门:如何上传与下载工程?

由于经常要在家写代码&#xff0c;所以需要有个能够方便访问代码管理工具。最近尝试了一下GitHub。经过了一翻纠结之后&#xff0c;基本上掌握了他的使用方式。 要使用GitHub需要首先在其网站上进行注册。其官方网站是https://github.com/。注册的流程在这里就不多少了&#x…

如何解决PIP命令不可用

今天想用PIP装一个python包&#xff0c;发现PIP报错&#xff0c;不是内部或外部命令。。。 遇事百度&#xff0c;有两种说法&#xff0c;一&#xff0c;没安装包&#xff0c;不管那么多命令执行了再说 在命令行输入&#xff1a;python -m ensurepip 将pip.exe文件下载下来 再pi…

RUNOOB python练习题5

用来练手的python 练习题其五&#xff0c;原链接 : python练习实例5 题干 : 输入三个整数x,y,z&#xff0c;请把这三个数由小到大输出。 又是非常简单的排序算法&#xff0c;只要使用numpy矩阵的排序方法或者使用python list的排序算法就可以轻松解决。 源代码如下 : import …

初步使用github,并上传下载文件

使用GitHub需要先注册GitHub的账号,登陆进去 然后开始创建项目 start a project 创建完成,开始生成公私钥,可以不必每次都要输密码 ssh-keygen -t rsa -C "mghxy123163.com" //填写email地址&#xff0c;然后一直“回车”ok 然后把公钥导入GitHub中的key里面去,也…

NOIP2000提高组复赛C 单词接龙

题目链接&#xff1a;https://ac.nowcoder.com/acm/contest/248/C 题目大意&#xff1a; 略 分析&#xff1a; 注意点&#xff1a;1.前缀和后缀的公共部分应该选最短的。2.如果两个字符串前缀和后缀的公共部分恰好是其中一个字符串&#xff0c;那么这两个字符串不能合并。 代码…