再来看一下我们的微服务架构 , 每个组件都需要服务器去部署,加起来可能需要几十个甚至上百个服务器。
这样的微服务项目在部署上会遇到什么问题?
问题:软件的迭代过程包括,设计,编码,编译,构建,测试,发布,运维等等流程,早期的软件开发模式为瀑布式开发,这种开发模式迭代更新太慢,每个环境都需要耗费大量人力和时间成本。往往很长时间才迭代一次。如今企业追求的都是敏捷开发:快速开发快速迭代,尽可能的缩短软件的开发生命周期。
DevOps 强调的是高效组织团队之间如何通过自动化的工具协作和沟通来完成软件的生命周期管理,从而更快、更频繁地交付更稳定的软件。
DevOps是一种方法或理念,它涵盖开发、测试、运维的整个过程。DevOps是提高软件开发、测试、运维、运营 等各部门的沟通与协作质量的方法和过程,DevOps强调软件开发人员与软件测试、软件运维、质量保障(QA)部 门之间有效的沟通与协作,强调通过自动化的方法去管理软件变更、软件集成,使软件从构建到测试、发布更加快 捷、可靠,最终按时交付软件。
DevOps是一套项目从开发到运维的解决方案,而不是具体的一个技术,DevOps需要整合一些列的技术来实现,比如:通过Git拉取代码,通过Mavn的命令进行项目的清理,编译,打包,测试等,通过Docker命令进行镜像构建…。
DevOps使用到的技术如:GitHub、Git/SVN、Docker、Jenkins、 Hudson、Ant/Maven/Gradle(编译,打包)、Selenium(自动化测试)、QUnit、JMeter(性能测试)等
下面普及2个概念
传统的瀑布式开发模型要求一个项目版本的所有功能都开发完成才会进行版本地迭代,这种开发模式不仅项目版本迭代速度慢,没法尽早的发现软件的质量问题,和持续的项目交付。
而敏捷开发最重要的目标是通过及早和持续不断地交付有价值的软件使客户满意;要达到这一目的就需要更高频率的版本迭代,包括:编译,打包,测试,运维。且应该自动化的完成,才有可能达到更快的软件开发生命周期。
敏捷与 DevOps 之间的主要区别在于:敏捷是关于如何开发和交付软件的哲学,而 DevOps 则描述了如何通过使用现代工具和自动化流程来持续部署代码。
云原生的定义最早也是由 Pivotal 公司的 Matt Stine 于 2013 年提出的,云原生是基于分布部署和统一运管的分布式云 [1] ,以容器、微服务、DevOps等技术为基础建立的一套云技术产品体系。 它具有:**DevOps、持续交付、微服务、容器、**等特点 。
云原生技术有利于各组织在公有云、私有云和混合云等新型动态环境中构建和运行可弹性扩展的应用。云原生的代表技术包括容器、服务网格、微服务、不可变基础设施和声明式 API。这些技术能够构建容错性好、易于管理和便于观察的松耦合系统。结合可靠的自动化手段,云原生技术使工程师能够轻松地对系统做出频繁和可预测的重大变更。
虚拟技术
在计算机中,虚拟化(英语:Virtualization)是一种资源管理技术,是将计算机的各种实体资源,如服务器、网络、内存及存储等,予以抽象、转换后呈现出来,打破实体结构间的不可切割的障碍,使用户可以比原本的组态更好的方式来应用这些资源,虚拟化技术主要用来解决高性能的物理硬件产能过剩和老的旧的硬件产能过低的重组重用
Docker
Docker 是一个开源的应用容器
引擎,让开发者可以打包
他们的应用以及依赖包到一个可移植的镜像
中,然后发布到任何流行的 Linux或Windows 机器上,也可以实现虚拟化。容器是完全使用沙箱机制,相互之间不会有任何接口
docker到特点:
镜像是构建Docker的基石。用户基于镜像来运行自己的容器。镜像也是Docker生命周期中的“构建”部分。镜像是基于联合文件系统的一种层式结构,由一系列指令一步一步构建出来
简单理解:安装系统需要一个IOS镜像,安装软件需要一个软件安装包,运行docker容器就需要对应的镜像。比如要启动一个Nginx容器,那么就需要从镜像仓库下载一个Docker的镜像然后再运行成容器。
Docker用Registry来保存用户构建的镜像。Registry分为公共和私有两种。Docker公司运营公共的Registry叫做Docker Hub。用户可以在Docker Hub注册账号,分享并保存自己的镜像(说明:在Docker Hub下载镜像巨慢,可以自己构建私有的Registry)。
docker本地存放镜像的仓库 , 远程下载的镜像就保存在本地仓库,如果:Maven的本地仓库
Docker可以帮助你构建和部署容器,你只需要把自己的应用程序或者服务打包放进容器即可。容器是基于镜像启动起来的,容器中可以运行一个或多个进程。 容器中运行的是项目或者软件,容器中有一个迷你的centos,容器需要进行才能运行
Docker是一个客户端-服务器(C/S)架构程序。Docker客户端只需要向Docker服务器或者守护进程发出请求,服务器或者守护进程将完成所有工作并返回结果。Docker提供了一个命令行工具Docker以及一整套RESTful API。你可以在同一台宿主机上运行Docker守护进程和客户端,也可以从本地的Docker客户端连接到运行在另一台宿主机上的远程Docker守护进程。
Docker官方建议在Ubuntu中安装,因为Docker是基于Ubuntu发布的,而且一般Docker出现的问题Ubuntu是最先更新或者打补丁的。在很多版本的CentOS中是不支持更新最新的一些补丁包的。 7最新版本
第一步:添加docker 的yum库
sudo yum install -y yum-utils device-mapper-persistent-data lvm2
sudo yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo
第二步:安装docker
sudo yum -y install docker-ce
第三步:启动docker
sudo systemctl start docker
第四步:镜像加速 :默认从国外镜像下载会比较慢,下面采用阿里云镜像库
vi /etc/docker/daemon.json
加入下面的内容
{
"registry-mirrors": ["https://5pfmrxk8.mirror.aliyuncs.com"]
}
下面是操作Docker的命令:systemctl命令是系统服务管理器指令它是 service和chkconfig两个命令组合。
镜像是构建Docker的基石。用户基于镜像来运行自己的容器。镜像也是Docker生命周期中的“构建”部分。镜像是基于联合文件系统的一种层式结构,由一系列指令一步一步构建出来.
docker images
这些镜像都是存储在Docker宿主机的/var/lib/docker目录下
使用名字:版本 和 使用id是一样的
docker rmi 镜像id/镜像名:版本
docker images -q
:删除所有镜像如果你需要从网络中查找需要的镜像,可以通过以下命令搜索
docker search 镜像名
注意:我们可以通过hub.docker.com 去查找镜像以及版本
国情的原因,国内下载 Docker HUB 官方的相关镜像比较慢,可以使用国内(docker.io)的一些镜像加速器,镜像保持和官方一致,关键是速度块,推荐使用。
docker pull 镜像名:版本
Docker可以帮助你构建和部署容器,你只需要把自己的应用程序或者服务打包放进容器即可。容器是基于镜像启动起来的,容器中可以运行一个或多个进程
创建容器常用的参数说明:
交互式容器示例: 交互式容器会提供一个伪终端
docker run -i -t --name=容器名字 -v=/外面目录:/里面目录 镜像:版本 /bin/bashsh
后台容器(守护式容器)
docker run -i -d --name=容器名字 -v=/外面目录:/里面目录 镜像:版本
注意:守护式容器不要-t 要 -d , 不要 /bin/bash
查看所有容器
docker ps -a
查看退出的容器
docker ps -f status=exited
查看最后一次运行的容器
docker ps – l
效果示例
有时候我们需要进入到容器中去查看容器的情况
docker exec -i -t 容器名字 /bin/bash
注意:进入容器需要 -t ,容器名字直接 ,需要 /bin/bash
暴力退出 : exit
注意:exit会结束交互式容器,而不会结束守护式容器 友好退出 : ctrl + p + q
kill 相比 stop 可以更加快速的关闭容器
docker stop 容器名/容器id 或 docker kill 容器名/容器id
停止中的容器可以通过start重新启动
docker start 容器名/容器id
请注意把容器和镜像的删除区分开,删除容器是 docker rm , 删除镜像是 docker rmi
docker rm 容器名/容器id
删除所有容器: docker ps -a -q 指的是获取所有容器的ID , 然后进行批量删除
docker rm `docker ps -a -q`
docker inspect 容器名
查看ip:容器是有自动分配的IP的,但是容器的IP重启之后会变化,所以一般不采用
docker inspect --format='{{.NetworkSettings.IPAddress}}' mycentos2
宿主机和容器之间进行文件共享可以通过-v 目录映射来实现,但是有的时候需要灵活的文件复制可以使用docker cp 来完成,如下
拷贝进去
docker cp 需要拷贝的文件或目录 容器名称:容器目录
拷贝出来
docker cp 容器名称:容器目录 需要拷贝的文件或目录
docker run -i -d --name=mysql -p=3306:3306 -e MYSQL_ROOT_PASSWORD=itsource123456 mysql:5.7
docker run -i -d --name=nginx -p=80:80 -v=/usr/local/nginx/html:/usr/share/nginx/html nginx
docker run -di --name=redis -p=6379:6379 redis --requirepass "123456"
容器由镜像启动而来,我们要安装某个软件就需要去下载相应的镜像,我们也可以自己去构建自己的镜像,构建镜像的方式包括如下三种
docker commit 容器名 新的镜像名:版本
Dockerfile是一个包含用于组合映像的命令的文本文档。可以使用在命令行中调用任何命令。 Docker通过读取Dockerfile中的指令自动生成映像。docker build命令用于从Dockerfile构建映像。可以在docker build命令中使用-f标志指向文件系统中任何位置的Dockerfile。Docker的指令如下
我们以构建JDK1.8的镜像为例,需求:使用Dockerfile , 基于centos:7 构建 jdk8的镜像
第二步: 创建一个目录,名字命名 :
mkdir /root/dockerfile
cd /root/dockerfile
第二步: 然后从window把jdk-8u171-linux-x64.tar.gz上传到 linux 到linux的 /root/dockerfile目录
第三步:vi Dockerfile ,内容如下
FROM centos:7
MAINTAINER itsource
WORKDIR /usr
RUN mkdir /usr/local/java
ADD jdk-8u171-linux-x64.tar.gz /usr/local/java
ENV JAVA_HOME /usr/local/java/jdk1.8.0_171
ENV JRE_HOME $JAVA_HOME/jre
ENV CLASSPATH $JAVA_HOME/bin/dt.jar:$JAVA_HOME/lib/tools.jar:$JRE_HOME/lib:$CLASSPATH
ENV PATH $JAVA_HOME/bin:$PATH
第五步:构建镜像,docker会去当前目录自动找名字为Dockerfile的脚本,然后一句一句执行脚本进行镜像构建
[root@VM-0-10-centos ~]# docker build -t="jdk:1.8" ./
使用Maven的Docker插件构建镜像的本质就是使用Dockerfile构建镜像,只不过Docker插件会自动帮我们创建Dockerfile脚本文件,它通过脚本命令,基于一个基础镜像,将我们的项目(项目.jar)拷贝进去打包形成一个新的镜像。
要做到如上功能需要开启Docker远程访问方便在开发机器上远程构建镜像。同时需要安装Maven的docker插件。
对于数量众多的微服务,手动部署(Dockerfile)无疑是非常麻烦的做法,并且容易出错。所以我们这里学习如何自动打镜像,这也是企业实际开发中经常使用的方法。
第一步:开启docker远程端口[危险]
[root@VM-0-10-centos ~]# vi /lib/systemd/system/docker.service
第二步:ExecStart=后添加配置 -H tcp://0.0.0.0:2375 -H unix://var/run/docker.sock
然后重新加载配置,重启Docker
[root@VM-0-10-centos ~]# systemctl daemon-reload //重新加载docker配置
[root@VM-0-10-centos ~]# systemctl restart docker //重启docker
开通防火墙
[root@VM-0-10-centos ~]# firewall-cmd --permanent --zone=public --add-port=2375/tcp
[root@VM-0-10-centos ~]# systemctl restart firewalld
访问测试 :服务器Ip:2375
使用docker插件的方式自动构建Dockerfile,然后基于docker构建镜像,直接把微服务打包成镜像推送到私服
<build>
<finalName>pethomefinalName>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
plugin>
<plugin>
<groupId>com.spotifygroupId>
<artifactId>docker-maven-pluginartifactId>
<version>0.4.13version>
<configuration>
<imageName>${project.artifactId}:${project.version}imageName>
<baseImage>jdk:1.8baseImage>
<entryPoint>["java", "-jar", "/${project.build.finalName}.jar"]entryPoint>
<resources>
<resource>
<targetPath>/targetPath>
<directory>${project.build.directory}
directory>
<include>${project.build.finalName}.jarinclude>
resource>
resources>
<dockerHost>http://ip:2375dockerHost>
configuration>
plugin>
plugins>
build>
使用idea的终端,进入到pom.xml所在目录,运行
mvn clean package -Dmaven.test.skip=true docker:build
注意:运行不成功的原因
不同的容器安装了不同的应用,应用之间需要通信,比如:项目所在的容器需要连接Redis容器,容器通信的方式也有多种放哪
容器之间默认可以使用容器的ip进行通信,但是重启docker,ip会变化。查看ip如下:
docker inspect 容器 | grep IPAddress
端口映射 把docker暴露到外部,外网通过 ip:映射的端口进行访问,有些应用是不应该暴露到外网的,比如: redis ,mysql就不应该暴露,但是为了操作方便也可以暴露,方便导入sql
启动容器的时候,给要通信的目标容器使用link指定一个“链接名”,在容器中就可以使用“链接名”和目标容器通信。格式: --link 目标容器:别名 ,例如: 某容器想要链接名字为 mysql 的容器,如下配置:
创建Mysql容器,名字为: mysql
Docker run -id --name=mysql -p=3306:3306 mysql:5.7
创建项目容器,名字为:pethome ,
docker run -di --name=myproject
--link mysql:mysql --link redis:redis pethome:1.0-SNAPSHOT
那么在项目中的配置文件中就可以这样来连接了
url: jdbc:mysql://mysql:3306/drive-order?serverTimezone=Asia/Shanghai&characterEncoding=utf8
还有一个种方式就是创建一个 桥接网络,如下:
docker network create -d bridge mybridge #创建自己的bridge
容器加入网络
docker run -d --name box5 --network mybridge busybox /bin/sh -c "while true;do sleep 3600;done"
docker run -d --name box6 --network mybridge busybox /bin/sh -c "while true;do sleep 3600;done"
运用自己的创建的bridge两个容器之间会自动link