玩过docker容器的人,对数据挂载肯定不陌生,volume :几种叫法,数据卷,资料卷
通过 -v 把宿主机目录绑定到容器中的目录
数据挂载的好处,仅仅是我认为的。不一定正确
1:数据能持久化保存,因为容器一旦删除,啥都没有了。但挂载的目录是不会删除的
2:修改和查看方便,比如要修改和查看数据,直接看本地,不用进去容器看,比较容器里面很多命令是没有安装的
这种方式是:-v 的方式称之为:bind mount,要区别于volume
volume方式是在Dockerfile 指定,比如:
VOLUME /data
VOLUME ["/data1","/data2"]
bind mount
容器以宿主机文件夹为准
volume
宿主有数据时,以宿主机为准
宿主无数据,从容器复制过来,再以宿主机为准
这里讲的是bind mount方式,通过查看容器的inspect 就可以发现
首先来 看一个我们常用的netcore 打包成镜像的Dockerfile
FROM mcr.microsoft.com/dotnet/core/aspnet:3.1-buster-slim AS base ENV TIMEZONE Asina/Shanghai WORKDIR /app EXPOSE 80 FROM mcr.microsoft.com/dotnet/core/sdk:3.1-buster AS build WORKDIR /src COPY . . RUN dotnet restore RUN dotnet build FROM build AS publish RUN dotnet publish "mytest.csproj" -c Release -o /app/publish FROM base AS final WORKDIR /app COPY --from=publish /app/publish . ENTRYPOINT ["dotnet", "mytest.dll"]
首先build成images
docker build -t test .
如果不挂载的方式,运行容器
docker run -d -p 8080:80 --restart=always --name my test
这样netcore文件都在容器的/app中,每次查看,都必须 docker exec ... 进入容器
那么已挂载的方式呢,相比很快就会想到了-v,立马尝试
docker run -d -p 8080:80 --restart=always -v /root/data:/app --name my test
-v /root/data:/app 意思是把容器内的app目录,挂载到宿主机的/root/data中,也就是2个目录bind
首先明确下,我这里是没有提前创建/root/data 这个目录的。不过挂载后,docker会给我们创建
但run之后,这个容器是起不来的,不过我这里加了restart,所以一直重启,
为什么这个容器run不起来?我们来分析下
只要理解了volume的意思,
bind 一定要注意,主机目录为空的话,会清空容器的目录,
也就是说,我们没挂载的时候,容器内app是有东西的,当我们把本地一个空 目录(/root/data)
挂载到容器内的/app下,就会清空容器内文件夹的内容,
就是说:你挂载了宿主机目录,我就以宿主机目录为主,容器的内有数据,我都给清空,所以导致容器无法启动
因为你Dockefile 中的入口是:ENTRYPOINT ["dotnet", "mytest.dll"]
都没有这个mytest.dll,自然无法启动
也许有人会问,压根就没必要这样挂载啊,我只挂载需要的,比如日志文件,配置文件,
那好,继续干
docker run -d -p 8080:80 --restart=always -v /root/logs:/app/logs -v /root/appsettings.json:/app/appsettings.json --name my test
logs 一开始肯定是为空,项目跑起来才生成,没问题
appsettings.json配置文件,手动拷贝即可
但我在开发的时候就有一个疑问,项目里面不仅仅这2个文件,还会有upload上传文件
template 模板文件,nlog 配置文件我觉得写多个-v不合适,
我就琢磨着。不能在容器启动的时候,把内部的数据拷贝到宿主机器吗。
容器数据拷贝到宿主机命令:
docker cp my:/app /root/myapp
宿主机拷贝到容器,反过来即可
docker cp /root/myapp my:/app
所以我想到了2种方法,估计不是最优的
1:数据挂载后,手动拷贝文件到本地目录,但我们是通过dockerfile 运行的
难道我又手动 dotnet publish ....一次,繁琐
2:容器运行后,把容器数据拷贝到本地,不行吗?是行的
分析:
既然数据在容器路径app中,运行的时候,会清空,那为什么不事先保存在其他目录中了
那修改下上面的Dockerfile
FROM mcr.microsoft.com/dotnet/core/aspnet:3.1-buster-slim AS base ENV TIMEZONE Asina/Shanghai WORKDIR /app EXPOSE 80 FROM mcr.microsoft.com/dotnet/core/sdk:3.1-buster AS build WORKDIR /src COPY . . RUN dotnet restore RUN dotnet build FROM build AS publish RUN dotnet publish "mytest.csproj" -c Release -o /app/publish FROM base AS final # 创建一个文件夹,存放文件,同时是被复制到宿主机的 RUN mkdir -p /data # 这是程序的最终工作目录,也就是宿主机挂载的目录 WORKDIR /app COPY --from=publish /app/publish /data ENTRYPOINT ["dotnet", "mytest.dll"]
加了一个data存放文件,程序启动的时候,把data拷贝到宿主机目录,绑定到宿主机app中
自己写一个shell文件,init.sh,我是这样写的
#运行容器,设置了restart,会一直重启,等待下面的拷贝完成,就会启动成功 docker run -d -p 8802:80 --restart=always -v /root/data:/app --name my test #把容器内部的数据拷贝到本地挂载目录 docker cp my:/data /root #删除容器中的data文件,非必须的操作 docker exec my rm -rf /data
然后执行 sh init.sh,OK,访问成功!
有没有感觉这样搞很麻烦。。确实,我自己看了都觉得麻烦,先是run,然后又cp。最后又exec
那为什么不把这些命令统一起来呢。仔细分析,上面的Dockerfile的入口是执行一个dll
ENTRYPOINT ["dotnet", "mytest.dll"]
当启动主程序之前还需要执行大量的前置操作时, 可以将 ENTRYPOINT 的入口指令设置为一个脚本 ,我这里添加一个 start.sh
所以我们改进下:
FROM mcr.microsoft.com/dotnet/core/aspnet:3.1-buster-slim AS base ENV TIMEZONE Asina/Shanghai WORKDIR /app EXPOSE 80 FROM mcr.microsoft.com/dotnet/core/sdk:3.1-buster AS build WORKDIR /src COPY . . RUN dotnet restore RUN dotnet build FROM build AS publish RUN dotnet publish "mytest.csproj" -c Release -o /app/publish FROM base AS final RUN mkdir -p /data WORKDIR /app COPY --from=publish /app/publish /data COPY start.sh /data RUN chmod +x /data/start.sh ENTRYPOINT ["/data/start.sh"]
我们依然是把publish 的文件放到data中。然后ENTRYPOINT 统一执行一个start.sh
start.sh脚本为:
#!/bin/bash mv -f /data/* /app dotnet mytest.dll
最后build和run就会按照预期的效果
docker build -t myimages . docker run -d -p 8804:80 -v /root/mynew:/app --name my myimages
然后在测试一次,依然成功,洗漱睡觉!!