本文簡要說一下ASP.NET Core 在Docker中部署以及docker-compose的使用 (ASP.NET Core 系列目錄), 系統環境為CentOS 8 ,
一、概述
簡單說一下Docker的幾個概念:
記得上學的時候流行一種安裝作業系統的方式,叫GHOST,大概是這樣的:

進入PE系統打開GHOST軟體,點擊“local”,然后選擇“Partition”,最后選擇“From Image”,選擇一個.gho后綴檔案,就開始系統安裝了,
安裝好系統之后,根據自己的需求又安裝了一些常用軟體,然后為了避免下次重裝系統還要安裝這些,可以將現在狀態的系統再次用GHOST備份一下,生成一個.gho后綴的鏡像檔案,這個鏡像又可以用來安裝系統,
一個.gho檔案可以用來為多臺電腦安裝系統,每個被安裝好的系統又可以被備份成一個.gho檔案檔案,
而類比Docker,有這樣幾個概念:
- Image(鏡像):有點像.gho后綴的鏡像檔案,
- Container(容器):就像用.gho安裝成功的一個作業系統,
- Repository(倉庫):存放鏡像的倉庫,像Git一樣可以有公有的倉庫也可以有私有的,微軟的倉庫地址為:
但實際上Docker不是一個作業系統,也不像一個虛擬機一樣,它是要共享宿主的內核的,
而且一般建議一個容器只跑一個行程,不像作業系統那樣可以多行程運行,(雖然也可以通過一些方法在一個Docker容器中跑多個應用,但不建議這樣做,)
二、安裝docker
說明:安裝CentOS 8 選擇了最小安裝,此處就不說了,下面說一下Docker的安裝程序,
- 安裝一些必要的系統工具:
sudo yum install -y yum-utils device-mapper-persistent-data lvm2
- 添加軟體源資訊:
sudo yum-config-manager --add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
- 更新 yum 快取:
sudo yum makecache fast
- 安裝 Docker-ce:
sudo yum -y install docker-ce
- 啟動 Docker 后臺服務
sudo systemctl start docker
注意:安裝Docker-ce的時候可能報錯:package docker-ce …… requires containerd.io >= 1.2.2-3, but none of the providers can be installed
是因為containerd.io版本過低,可去下面網站查看新版本:
https://download.docker.com/linux/centos/7/x86_64/edge/Packages
- 下載:
wget https://download.docker.com/linux/centos/7/x86_64/edge/Packages/containerd.io-1.2.6-3.3.el7.x86_64.rpm
- 安裝:
yum -y install containerd.io-1.2.6-3.3.el7.x86_64.rpm
再次執行sudo yum -y install docker-ce安裝即可,
三、Docker的幾個常見命令
- 搜索遠程存盤庫中的鏡像,例如MongoDB的鏡像
docker search mongo

- 拉取倉庫中的鏡像
docker pull mongo
- 列出本地鏡像,
docker images
可以看到本地鏡像中包了mongo鏡像,
- 運行鏡像生成一個容器
docker run --name mongotodocker -p 27088:27017 -d mongo
含義: 用鏡像mongo運行生成一個容器,名字為mongotodocker ,將容器內的埠27017映射到主機的27088埠,-p 指的是埠映射, -d是說后臺運行容器,并回傳容器ID;
- 列出所有容器,
docker ps -a
可以看到剛運行起來的容器,
- 停止容器
docker stop mongotodocker
- 洗掉容器,
docker rm mongotodocker
- 洗掉鏡像
docker rmi mongo
具體每個命令都有一些引數可用,這里只是簡單介紹一下使用方法,具體的檔案網上很多,不一一說明了,
四、注冊Docker賬號
注冊一個賬號(可選項),地址:https://hub.docker.com/ ,可以在上面建自己的倉庫,
五、創建一個ASP.NET Core 專案,生成并運行Docker鏡像
新建一個名為DockerComposeDemo的API專案,直接發布,拷貝發布的檔案到CentOS系統中,例如/home/aspcore目錄,并在該目錄新建一個文本檔案名為Dockerfile,內容如下:
FROM mcr.microsoft.com/dotnet/core/aspnet:3.0-buster-slim AS base
WORKDIR /app
EXPOSE 80
COPY . .
ENTRYPOINT ["dotnet", "DockerComposeDemo.dll"]
含義是:參考包含3.0運行時的鏡像,這個鏡像在遠程倉庫中,若本地沒有提前pull下來,會先執行pull操作獲取到本地,然后將作業目錄設為/app , 拷貝發布的專案檔案,設定行程的入口是通過dotnet運行DockerComposeDemo.dll,
執行如下命令:
cd /home/aspcore
docker build -t dockertest .
注意第二行后后面有個'.'不能少, 含義就是按照Dockerfile檔案中設定的規則生成名為dockertest的鏡像,
此時執行docker images命令可以看到本地鏡像中已經有了 mcr.microsoft.com/dotnet/core/aspnet:3.0-buster-slim 和 dockertest 兩個鏡像,
運行這個鏡像生成容器:
docker run --name aspdocker -p 8080:80 -d dockertest
生成一個名為aspdocker 的容器,并將容器的80埠映射到主機的8080埠,訪問專案默認提供的controller:http://192.168.183.230:8080/WeatherForecast
可以看到能正常訪問,
六:使用docker-compose
因為一個Docker容器只建議運行一個應用,那么一個專案就可能會存在多個容器被運行,可能包含多個專案、資料庫等,這時候就需要對這些容器進行統一的管理,從構建運行開始到運行后狀態的監控等,
這時候有個簡易的方法就是docker-compose,它可以完成多個Docker的統一管理,包括Docker鏡像構建、容器運行、相關配置以及Docker之間的依賴關系等,
下面舉個簡單例子,這個DockerComposeDemo專案需要搭配一個MongoDB資料庫,這樣除了該專案外還需要一個Docker容器運行MongoDB資料庫,
這時候用docker-compose就方便多了,docker-compose的核心是docker-compose.yml檔案,看一下對應這個例子的檔案內容:
version: '3.4'
services:
demomvc:
image: thisdemoimage
build:
context: .
dockerfile: Dockerfile
environment:
- ASPNETCORE_DBCONN=mongodb://192.168.183.230:27089
- ASPNETCORE_DBNAME=dockerdb
ports:
- "5103:80"
depends_on:
- mongodocker
mongodocker:
image: mongo
ports:
- "27089:27017"
在services節點下定義了demomvc和mongodocker兩個服務,一個是ASP.NET Core的專案,一個是MongoDB資料庫,
每個節點下的image引數指定了采用的鏡像名稱,ports指定埠映射,此處的MongoDB設定未涉及持久化,實際使用時要注意設定,
ASP.NET Core的專案的thisdemoimage鏡像是不存在的,下面指定了build方法,當然也可以先創建好鏡像然后在這里使用就像mongo服務的設定一樣,
depends_on表示本服務對另一個服務的依賴,本例中就是ASP.NET Core專案依賴MongoDB專案,
environment用于設定環境變數,作用是什么呢?
有一些設定,比如本例中的資料庫連接,如果將連接字串寫在了專案中的appsettings.json中,而這個檔案被“固化”到鏡像中了,是不能修改的,除非重新生成鏡像,非常麻煩,
所以可以通過這樣的環境變數在外面設定,
將專案參考NuGet包MongoDB.Driver, 修改WeatherForecastController的get方法:
[HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); _mongoHelper.InsertOne(new WeatherForecast { Date = DateTime.Now.AddDays(1), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }); return _mongoHelper.FindList<WeatherForecast>(); }
每次都是先插入一條,然后回傳所有記錄,這里簡要的寫了一個mongoHelper:
public class MongoHelper { private readonly IMongoDatabase database; public MongoHelper(IConfiguration configuration) : this(configuration["ASPNETCORE_DBCONN"], configuration.GetSection("ASPNETCORE_DBNAME").Value) { } public MongoHelper(string ConnectionString,string DBName) { MongoClient mongoClient = new MongoClient(ConnectionString); database = mongoClient.GetDatabase(DBName); } public List<T> FindList<T>(FilterDefinition<T> filter = null, string collectionName = null) { collectionName ??= typeof(T).Name; filter ??= new BsonDocument(); var collection = database.GetCollection<T>(collectionName); return collection.Find(filter).ToList(); } public void InsertOne<T>(T model, string collectionName = null) { collectionName ??= typeof(T).Name; var collection = database.GetCollection<T>(collectionName); collection.InsertOne(model); } }
連接字串采用 IConfiguration中的設定,
這里有個不算技巧的技巧,為了方便在非Docker的情況下測驗,依然可以在appsettings.json檔案中設定MongoDB的連接字串,當部署到Docker中的時候,通過Docker環境變數配置的連接字串會覆寫appsettings.json中的配置,
這是因為在講述IConfiguration的文章中說過,系統是先加載appsettings.json中的設定,后加載環境變數中的設定的,二者的key相同,所以最侄訓以環境變數中的配置為準,
重新發布專案并將檔案拷貝到/home/aspcore目錄,其中的dockerfile檔案不變,添加本例中的docker-compose.yml檔案,
docker-compose是需要單獨下載安裝的, 執行命令:
sudo curl -L "https://github.com/docker/compose/releases/download/1.24.1/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
如果提示權限錯誤,需執行如下命令:
sudo chmod +x /usr/local/bin/docker-compose
安裝好之后執行 docker-compose --version 驗證是否安裝成功,
都準備好了,執行如下命令:
cd /home/aspcore
docker-compose up
執行成功后訪問 http://192.168.183.230:5103/WeatherForecast 進行測驗,
七、Windows下開發
我們都知道,VisualStudio經常“貼心”的幫我們做好多事,例如Git的圖形化操作,對于Docker也是如此,
若要在Windows環境下開發及除錯Docker,可按下面步驟完成,
首先需下載并安裝Docker Desktop
頁面上有個圖示:
,點擊下載,安裝
后右下角會有 圖示,右鍵可以做一些設定,
它支持Windows和Linux兩種主機
通過docker version 命令可以看出當前主機型別,也可以右鍵點擊右下角的圖示,有個Switch to ……的選項,可以知道當前主機型別,點擊后切換到另一種型別,
命令切換:C:\Program Files\Docker\Docker\DockerCli.exe -SwitchDaemon
解決方案啟用Docker支持:
新建專案的時候,勾選啟用Docker支持:

已有專案可以右鍵點擊專案,添加Docker支持:

兩種方式都會要求選擇主機型別是Windows還是Linux,
此時Visual Studio幫我們會在專案中添加一個名為Dockerfile的檔案:
FROM mcr.microsoft.com/dotnet/core/aspnet:3.0-buster-slim AS base WORKDIR /app EXPOSE 80 FROM mcr.microsoft.com/dotnet/core/sdk:3.0-buster AS build WORKDIR /src COPY ["DockerDemo/DockerDemo.csproj", "DockerDemo/"] RUN dotnet restore "DockerDemo/DockerDemo.csproj" COPY . . WORKDIR "/src/DockerDemo" RUN dotnet build "DockerDemo.csproj" -c Release -o /app/build FROM build AS publish RUN dotnet publish "DockerDemo.csproj" -c Release -o /app/publish FROM base AS final WORKDIR /app COPY --from=publish /app/publish . ENTRYPOINT ["dotnet", "DockerDemo.dll"]
這個檔案和上面例子中我們自己創建的優點不同,它包含了4個From,第一個和最后一個和我們自己創建的有點像,只是Visual Studio幫我們自動添加了SDK鏡像的拉取、專案的編譯、專案發布的程序,
這里用到了兩個鏡像,第一個From呼叫了微軟官方的包含ASP.NET Core 3.0 的運行時版鏡像,第二個From用到了包含.Net Core 3.0的SDK的鏡像,因為我們需要對專案進行生成和發布操作,
通過添加Docker的支持,可以使用Visual Studio開發并將專案自動發布到Docker進行除錯,但選擇系統環境為Windows的時候速度很快,選擇Linux的時候由于網路問題非常慢,網上有臨時的解決方案,
如果多個專案想采用docker-compose管理,在上面添加docker支持的圖中可以看到有一個“容器業務流程協調程式支持”, 添加它就會自動生成一個docker-compose.yml檔案,
Docker-Compose主要用于當前主機中的docker的管理,對于多主機的集群管理,就需要Docker Swarm或者Kubernetes了,
轉載請註明出處,本文鏈接:https://www.uj5u.com/net/98028.html
標籤:.NET Core
