前言
在Docker生态系统中除了上一节所讲解的基本概念,还有其他专业术语,本文我们将一笔带过,同时会开始陆续进入到在.NET Core中使用Docker。
专业术语
Docker Engine(Docker引擎):客户端 - 服务器应用程序。 Docker公司将Docker Engine分为两个产品。 Docker Community Edition(CE)基于开源工具且免费,我们学习时可以使用这款产品。 Docker Enterprise附带了其他功能支持,比如管理和安全等等功能。
Docker Client(Docker客户端):我们与Docker进行交互的主要方式。 使用Docker命令行界面(CLI)时,在终端中键入以docker开头的命令,Docker Client会使用Docker API将命令发送到Docker守护进程中。
Docker Daemon(Docker守护进程):监听Docker API请求的Docker服务器。 Docker守护进程主要管理镜像,容器,网络和卷。
Docker Volumes(Docker卷):存储创建应用程序和运行应用程序持久化数据的最佳方式。
Docker Registry:存储Docker镜像的远程注册位置。 我们将镜像推送到注册表并从注册表中提取图像,我们可以托管自己的注册表或使用供应商商的注册表。
Docker Hub:Docker镜像的最大注册表。 它也是Dcoker默认的注册表位置, 我们可以在Docker Hub上找到免费的镜像并存储我们自己的图像。
Docker Networking:允许我们将Docker容器连接在一起。 连接的Docker容器可以位于同一主机或多个主机上。
Docker Compose:属于一个工具,我们可以非常轻松地运行需要多个Docker容器的应用程序。 Docker Compose允许我们将命令写到docker-compose.yml文件中以供重用。 Docker Compose命令行界面(cli)使我们可以更轻松地与多容器应用程序进行交互。 Docker Compose免费安装Docker。
Docker Swarm:容器部署编排的产品。Docker官方教程进行容器编排和部署使用的Docker Swarm。 建议不要浪费时间在Docker Swarm上,推荐使用Kubernetes(k8s)。
Docker Services:分布式应用程序的不同部分。 服务实际上只是“生产中的容器”。服务只运行一个镜像,但它编码了镜像的运行方式 - 它应该使用哪些端口,容器应该运行多少个副本,以便服务具有所需的容量等等。 扩展服务会更改运行该软件的容器实例的数量,从而为流程中的服务分配更多计算资源。Docker服务允许我们跨多个Docker守护进程扩展容器,并使Docker Swarms成为可能。
回顾容器
Docker镜像在构建时被创建,而Dokcer容器在运行时被创建。Dockerfile是Docker的核心,Dockerfile告诉Docker如何构建镜像从而被用来制作容器,每个Docker镜像都包含一个名为Dockerfile但没有扩展名的文件。当调用docker build以创建映像时,假定Dockerfile位于当前工作目录中,可以使用文件标志(-f)指定到其他位置,容器是由一系列层所构建,而且每个镜像层只读,除了位于其他镜像层之上的最终容器镜像层。 Dockerfile告诉Docker要添加哪些层以及添加它们的顺序,每个镜像层实际上只是一个包含自上一层以来的所更改的文件。 在linux中,几乎所有东西都是文件。基础镜像提供初始层,基础镜像也称为父镜像,将镜像从远程存储库提取到本地时,仅仅只下载本地计算机上尚未存在的层, Docker通过重用现有层来节省空间和时间。
Dockerfile指令是一行开头的大写单词,后紧跟其参数,Dockerfile中的每一行都可以包含一条指令。 构建图像时,将从上到下处理指令,如下:
只有FROM,RUN,COPY和ADD指令才能在最终镜像中创建镜像层,其他指令只是作为配置或说明,比如添加元数据或告诉Docker在运行时执行某些操作,例如公开端口或运行命令。在本文中,我们使用基于linux的Docker镜像,当然我们也可以使用基于Windows的镜像,建议使用linux。接下来我们来过滤下Dockerfile中各个指令说明。
Dockerfile指令
FROM - 指定基础(父)镜像。
LABEL - 提供元数据,包含维护者信息。
ENV - 设置持久化环境变量。
RUN - 运行命令并创建镜像层,用于将包安装到容器中。
COPY - 将文件和目录复制到容器中。
ADD - 将文件和目录复制到容器中。 可以upack本地.tar文件。
CMD - 为执行容器提供命令和参数,可以覆盖参数,只能有一个CMD。
WORKDIR - 设置后续说明的工作目录。
ARG - 定义一个在构建时传递给Docker的变量。
ENTRYPOINT - 为正在执行的容器提供命令和参数。
EXPOSE - 对外暴露端口。
VOLUME - 创建目录用于访问和存储持久化数据。
.NET Core入门例子
接下来我们以.NET Core中使用Docker并输出Hello World结束本文。
我们通过命令创建一个.NET Core控制台程序,接下来为了在页面上输出Hello World,我们需要使用中间件,所以我们添加AspNetCore包,如下:
然后我们打开控制台程序,添加中间件打印Hello World代码:
public class Startup { public void Configure(IApplicationBuilder applicationBuilder, IHostingEnvironment hostingEnvironment) { applicationBuilder.Run(async context => { await context.Response.WriteAsync("Hello World"); }); } }
class Program { static void Main(string[] args) { WebHost.CreateDefaultBuilder() .UseStartup<Startup>() .UseKestrel() .UseUrls("http://0.0.0.0:5050") .Build() .Run(); } }
程序已就绪完毕,接下来我们发布该控制台程序,如下:
接下来将执行上述步骤生成的bin目录(实际上只需拷贝发布后生成的publish目录即可,为了省事,我直接拷贝了整个bin目录)拷贝到ubuntu中(由于我对linux不熟悉,所以采用虚拟机加载桌面端ubuntu镜像的方式,对于从未使用过linux的童鞋,推荐使用桌面端ubuntu,友好的GUI界面,方便我们初学知道各个文件夹是做什么的,一看便知,后续再使用服务端版的ubuntu就会得心应手啦)。
接下来我们进入HelloWorld文件目录,创建Dockerfile文件从而来创建.NET Core镜像。
然后我们来编写Dockerfile文件构建镜像,如下:
FROM mcr.microsoft.com/dotnet/core/aspnet:2.2 AS runtime COPY ./bin/Debug/netcoreapp2.2/linux-x64/publish/ ./ ENTRYPOINT ["dotnet", "HelloWorld.dll"]
父镜像为.NET Core 2.2版本和我们创建的程序版本一致,然后将我们的应用程序(即publish目录)拷贝,最后指定程序运行的命令和参数。有了Dockerfile文件,那么我们就可以开始构建镜像了,终端继续运行如下命令(注意:镜像标签名称必须全部为小写,否则报错):
docker build . -t hellowrold
镜像已构建完毕,接下来则是创建并启动容器运行程序,如下:
docker run -p 5050:5050 hellowrold
从上述我们可看到容器已正常启动,且运行环境为生产环境,监听端口为5050。桌面版ubuntu默认为我们安装了火狐浏览器,此时我们打开浏览器将会输出Hello World,如下:
总结
本文我们介绍Docker中的一些术语,然后最后写了一个在.NET Core中使用Docker的入门例子,非常简单。若直接使用服务端版本的ubuntu我是一脸懵逼,有了界面,我也大概知道了一些文件夹里存放的是什么,一目了然,虽说这还只是冰山一角,每天积累一点,日积月累,厚积薄发嘛不是。