利用Docker分層構(gòu)建優(yōu)化鏡像大小的實(shí)現(xiàn)
Docker 鏡像大小優(yōu)化的重要性
資源利用效率
較小的鏡像文件在存儲(chǔ)和傳輸過(guò)程中占用更少的空間和帶寬。例如,在將鏡像從本地倉(cāng)庫(kù)推送到遠(yuǎn)程倉(cāng)庫(kù)或者在集群環(huán)境中分發(fā)鏡像時(shí),小鏡像可以加快傳輸速度,減少網(wǎng)絡(luò)資源的占用。
當(dāng)在主機(jī)上存儲(chǔ)大量鏡像時(shí),較小的鏡像可以節(jié)省磁盤空間,使得可以在有限的磁盤容量下存儲(chǔ)更多的鏡像,這對(duì)于容器編排系統(tǒng)(如 Kubernetes)管理大量不同的容器鏡像非常重要。
容器啟動(dòng)速度
小鏡像啟動(dòng)速度通常更快。因?yàn)樵谌萜鲉?dòng)時(shí),需要將鏡像中的文件系統(tǒng)加載到容器的存儲(chǔ)層,如果鏡像文件較小,那么加載的時(shí)間和資源消耗會(huì)相應(yīng)減少,從而提高容器的啟動(dòng)效率。
分層構(gòu)建鏡像的原理和優(yōu)勢(shì)
原理
Docker 鏡像是由多層文件系統(tǒng)疊加而成的。每一層都代表了一次對(duì)文件系統(tǒng)的修改,例如安裝軟件包、添加配置文件等操作。當(dāng)構(gòu)建鏡像時(shí),Docker 會(huì)將這些操作按順序記錄下來(lái),形成一個(gè)分層的結(jié)構(gòu)。
例如,基礎(chǔ)鏡像(如ubuntu:latest
)是最底層,然后在其上添加新的層,如安裝nginx
軟件包會(huì)形成一個(gè)新的層,修改nginx
的配置文件又會(huì)形成另外一層。
優(yōu)勢(shì)對(duì)鏡像大小的影響
復(fù)用已有層:分層構(gòu)建最大的優(yōu)勢(shì)之一是可以復(fù)用層。如果多個(gè)鏡像都基于相同的基礎(chǔ)鏡像,并且在基礎(chǔ)鏡像之上的某些層是相同的(例如都安裝了相同的系統(tǒng)庫(kù)),那么這些相同的層在存儲(chǔ)時(shí)只會(huì)保存一份。
例如,有鏡像 A 和鏡像 B 都基于
ubuntu:latest
構(gòu)建,并且都安裝了python3 - pip
包,那么這個(gè)python3 - pip
安裝層只會(huì)在存儲(chǔ)中保存一次,多個(gè)鏡像可以共享該層。這樣可以大大減少鏡像存儲(chǔ)所需的空間總量。增量構(gòu)建和更新:當(dāng)需要更新鏡像中的某一部分時(shí),只需要更新對(duì)應(yīng)的層。例如,如果只是更新了應(yīng)用程序的代碼而沒有改變基礎(chǔ)操作系統(tǒng)或其他依賴庫(kù),那么只有包含代碼的層會(huì)被更新,其他層可以保持不變。這使得更新鏡像時(shí)的數(shù)據(jù)傳輸量最小化,有助于保持鏡像大小的合理性。
分層構(gòu)建鏡像的最佳實(shí)踐
合理選擇基礎(chǔ)鏡像
基礎(chǔ)鏡像的大小對(duì)最終鏡像大小有很大影響。盡量選擇輕量級(jí)的基礎(chǔ)鏡像。例如,對(duì)于基于 Linux 的應(yīng)用程序,如果不需要完整的 Ubuntu 或 CentOS 系統(tǒng)的所有功能,可以選擇
alpine
作為基礎(chǔ)鏡像。alpine
是一個(gè)輕量級(jí)的 Linux 發(fā)行版,其鏡像大小通常只有幾兆字節(jié),相比完整的 Ubuntu 或 CentOS 鏡像(可能幾百兆字節(jié))要小很多。例如,構(gòu)建一個(gè)簡(jiǎn)單的 Python Web 應(yīng)用容器,使用
python:alpine
作為基礎(chǔ)鏡像會(huì)比使用python:ubuntu
產(chǎn)生更小的鏡像。合并多層操作(減少層數(shù))
雖然分層有很多優(yōu)勢(shì),但過(guò)多的層也可能導(dǎo)致一些問(wèn)題,如鏡像存儲(chǔ)和傳輸效率下降。在構(gòu)建鏡像過(guò)程中,可以將一些相關(guān)的操作合并到同一層中。
例如,在安裝多個(gè)軟件包時(shí),可以在一個(gè)
RUN
指令中完成,而不是為每個(gè)軟件包安裝使用一個(gè)單獨(dú)的RUN
指令。比如在Dockerfile
中,不要這樣寫:
RUN apt - get update RUN apt - get install - y package1 RUN apt - get install - y package2
應(yīng)該這樣寫:
RUN apt - get update && apt - get install - y package1 package2
這樣可以減少鏡像的層數(shù),因?yàn)槊恳粋€(gè)RUN
指令都會(huì)產(chǎn)生一個(gè)新的層。
- 清理無(wú)用文件和緩存
在構(gòu)建鏡像過(guò)程中,安裝軟件包或編譯程序等操作可能會(huì)產(chǎn)生一些臨時(shí)文件和緩存。這些文件會(huì)增加鏡像的大小,但在容器運(yùn)行時(shí)可能并不需要。
例如,在apt - get
安裝軟件包后,可以使用apt - get clean
命令清理軟件包緩存。在構(gòu)建基于 Java 的應(yīng)用程序鏡像時(shí),編譯后的字節(jié)碼文件可能不需要包含源代碼文件,那么可以在構(gòu)建過(guò)程中刪除源代碼文件。在Dockerfile
中可以這樣寫:
RUN apt - get update && apt - get install - y package && apt - get clean
或者在構(gòu)建 Java 應(yīng)用程序鏡像時(shí):
# 假設(shè)編譯后的代碼在/target目錄,源代碼在/src目錄 COPY. /app WORKDIR /app RUN mvn clean install - DskipTests RUN rm - rf /app/src
通過(guò)分層構(gòu)建鏡像并結(jié)合上述最佳實(shí)踐,可以有效地控制 Docker 鏡像的大小,提高鏡像的存儲(chǔ)和使用效率。
- 多階段構(gòu)建
原理:多階段構(gòu)建允許將構(gòu)建過(guò)程分為多個(gè)階段,每個(gè)階段可以使用不同的基礎(chǔ)鏡像和構(gòu)建工具。在前面的階段可以進(jìn)行代碼編譯、測(cè)試等操作,在后面的階段只復(fù)制運(yùn)行時(shí)需要的文件,從而減少最終鏡像的大小。
示例:以一個(gè) Go 語(yǔ)言應(yīng)用為例,第一階段可以使用包含 Go 編譯器的基礎(chǔ)鏡像來(lái)編譯應(yīng)用程序,第二階段可以使用一個(gè)輕量級(jí)的alpine
基礎(chǔ)鏡像來(lái)運(yùn)行編譯后的程序。
# 第一階段:構(gòu)建應(yīng)用 FROM golang:1.18 - alpine AS builder COPY. /app WORKDIR /app RUN go build - o myapp # 第二階段:運(yùn)行應(yīng)用 FROM alpine:latest COPY -- from = builder /app/myapp /usr/local/bin/myapp CMD ["myapp"]
在這個(gè)例子中,第一階段構(gòu)建出了 Go 應(yīng)用,第二階段只復(fù)制了運(yùn)行時(shí)需要的可執(zhí)行文件myapp
到一個(gè)輕量級(jí)的alpine
鏡像中,最終的鏡像不包含 Go 編譯器和其他構(gòu)建過(guò)程中產(chǎn)生的不必要的文件,大大減小了鏡像大小。
以上就是利用Docker分層構(gòu)建優(yōu)化鏡像大小的實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于Docker分層構(gòu)建鏡像的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Docker容器搭建android編譯環(huán)境的實(shí)踐記錄
這篇文章主要介紹了Docker容器搭建android編譯環(huán)境的實(shí)踐記錄,主要包括部署容器、鏡像管理、容器管理等相關(guān)知識(shí),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07使用Docker構(gòu)建企業(yè)級(jí)自定義鏡像的方法
這篇文章主要介紹了使用Docker構(gòu)建企業(yè)級(jí)自定義鏡像的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05云centos開啟docker遠(yuǎn)程服務(wù)鏈接的實(shí)現(xiàn)步驟
本文主要介紹了云centos開啟docker遠(yuǎn)程服務(wù)鏈接,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-09-09Deepin使用docker安裝mysql數(shù)據(jù)庫(kù)過(guò)程詳解
這篇文章主要介紹了Deepin使用docker安裝mysql數(shù)據(jù)庫(kù)過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06