網站首頁 編程語言 正文
前言
回想上學的時候,某老師神采奕奕的講解著操作系統某個知識點原理,可謂是激情澎湃,我環顧周圍基友們,臉上懵逼兩字清晰可見,毫不含糊,恍然大悟原來是在講天書。為了不讓大家臉上呈現懵逼二字,希望能以人能看懂的語言,讓大家理解。
一、Docker run
這一小節,標題是Docker run ,我想受到過九年義務教育的大家,不難翻譯吧,如果這都翻譯不了,那么你一定要順著網線爬過來找我,我一定把你腦殼打得開竅。
上一章節,最后我們用Docker啟動了一個ningx的容器:
#運行容器 docker run --name nginx-container -p 80:80 -d nginx
瀏覽器輸入虛擬機地址即可訪問 Nginx
那么,docker run啟動命令是干了什么,才生成了一個nginx容器的呢?帶著疑問,我們接著說。
首先我們要很明白 運行容器的命令
nginx-contarner
:nginx容器的名稱(容器相當于java中類生成的對象)nginx
:鏡像(鏡像相當于java中類)
(下圖借鑒一位博主的,如有不合適,請我聯系我刪除)
運行容器命令,做了什么呢?首先Docker會去本機尋找鏡像,如果沒有找到,它會去DockerHub上去下載,如果能夠找到并下載到本地就會使用這個鏡像去執行,否則就會返回找不到該鏡像的錯誤。
就好比,你在家里上廁所,需要紙,你就會在廁所里尋找衛生紙,如果找到了,就直接可以用了。但是,如果你沒有找到,那就得叫老爸在家里找衛生紙,找到了你就可以安心笑了,沒有找到的話,老爸就會告訴你家里沒有衛生紙紙。這里的衛生紙就是所謂的鏡像.
二、Docker底層原理
1、Docker是怎么工作的
我們知道Docker是一個客戶端-服務端(C/S)架構。根據上節講述的C/S架構運行模式就不難去理解。
Docker客戶端只需要向Docker守護進程發出請求,Docker守護進程在接收到Docker客戶端的請求后去執行針對容器的相應操作,完成所有工作后返回結果。(詳細深入待后續)
守護進程和普通進程區別是指 : 將后臺程序變成一種服務,比如說,用命令行輸入啟動程序,如果不是守護進程的話,一旦命令行窗口關閉,程序就終止了;而如果啟動守護進程,則退出命令行窗口之后,服務一直處于運行狀態。
2、為什么Docker比虛擬機快
Docker與虛擬機的對比,第一節文章就已經闡述過了。通過對比,我們就可以知道為什么Docker比虛擬機快。這里我在總結下:
1、容器由于沒有了虛擬操作系統和虛擬機監視器這兩個層次,大幅減少了應用程序運行帶來的額外消耗。
2、Docker利用的是宿主機的內核,那么容器中的應用程序就是完全運行在了宿主操作系統中。因此在docker容器里運行應用程序,與在宿主機中運行應用程序效果是一樣的。
三、Docker鏡像
1、鏡像
鏡像是Docker容器的基石,容器是鏡像的運行實例,有了鏡像才能啟動容器。它包含運行某個軟件所需的所有內容,包括代碼、運行時、庫、環境變量和配置文件。(這里可以去上一篇文章中,有對Docker鏡像的形象比喻)
2、聯合文件系統UnionFS
要想理解Docker鏡像的加載原理,那么我們得要知道UnionFS(聯合文件系統)
1、Union文件系統(UnionFS) 是一種分層、輕量級并且高性能的文件系統,它支持對文件系統的修改作為一次提交來一層層的疊加,同時可以將不同目錄掛載到同一個虛擬文件系統下。Union文件系統是 Docker 鏡像的基礎。鏡像可以通過分層來進行繼承,基于基礎鏡像(沒有父鏡像),可以制作各種具體的應用鏡像。
看這文縐縐的Union文件系統的描述,難免呈現懵逼二字,過于抽象。看著都沒耐心繼續往下看…理解不了
UnionFS 最主要的功能簡單理解為把不同的目錄合并成一個目錄,原來目錄下的內容路徑都不會改變。比如,我現在有兩個目錄 A 和 B,它們分別有兩個文件:
$ tree . ├── A │ ├── a │ └── x └── B ├── b └── x
然后,使用聯合掛載的方式,將這兩個目錄掛載到一個公共的目錄 C 上:
$ mkdir C $ mount -t aufs -o dirs=./A:./B none ./C
這時,再查看目錄 C 的內容,就能看到目錄 A 和 B 下的文件被合并到了一起:
$ tree ./C ./C ├── a ├── b └── x
可以看到,在這個合并后的目錄 C 里,有 a、b、x 三個文件,并且 x 文件只有一份,同時合并后的目錄對于原來的目錄里文件的路徑是不會改變的。這,就是“合并”的含義。
但是,大家應該會有個疑問,就是x文件到底是原先哪個目錄下的。答案是誰先被掛載就顯示成誰的,也就是說如果目錄A先被掛載,那優先顯示目錄A里的x文件。(具體詳細深入后面講解)
3、Docker鏡像的加載原理
Docker設計時,就充分利用Union FS的技術,將其設計為分層存儲的架構。 鏡像實際是由多層文件系統聯合組成,這種層級的文件系統UnionFS,在內部又分為2部分:
1) bootfs(boot file system):docker鏡像的最底層是bootfs,主要包含bootloader(加載器)和kernel(內核)。bootloader主要是引導加載kernel,linux剛啟動時會加載bootfs文件系統。當bootfs加載完成后,整個內核就在內存中了,此時內存的使用權已由bootfs轉交給了內核,此時系統也會卸載bootfs。
這里的加載,可以理解為,我們windows電腦開機時候,從黑屏到進入操作系統的過程。
2)rootfs(root file system):在bootfs之上,包含的就是典型linux系統中的/dev、/proc、/bin、/etc等標準目錄和文件。不同的 Linux 發行版, boots 基本是一致的, rootfs 會有差別。
舉個例子:如下圖所示
以圖為例,從左到右,分為3個過程:
1、docker鏡像的最底層是bootfs,然后是一個Base Image的基礎鏡像,這個基礎鏡像是Centos。
2、執行安裝mysql操作,那么Docker會在Centos基礎鏡像之上又加了一層mysql鏡像。
3、執行安裝tomcat,那么在之前的mysql鏡像上在繼續加一層tomcat鏡像。
就像疊積目一樣,一層一層往上。也說明了docker的鏡像實際上是由層一層的文件系統組成的。對于不同的的linux發行版本,bootfs基本是一致的,rootfs會有差別,所以不同的發行版可以共用bootfs。
另外,平時我們安裝進虛擬機的CentOS都是好幾個G,為什么Docker這里才200M?
因為底層直接用主機的內核,自己只需要提供rootfs就行了,所以rootfs可以很小,只需要包含最基本的命令、工具和程序庫即可。這樣一來,啟動速度也快了,因為最浪費時間的引導加載過程沒了。
4、Docker鏡像分層理解
知道了鏡像的加載原理,我們在來看看Docker鏡像的分層理解。
首先可以去下載一個reids 鏡像,注意觀察下載的日志輸出,可以看到是一層層的在下載。
當下載的層文件與我們之前層文件有沖突,也就是下載過的文件就會顯示Already exists不會再去下載,它只會去下載一些跟 redis 相關的新的東西。
那么:為什么Docker鏡像要采用這種分層的結構呢?
這種方式最大的好處就在于資源共享。比如有多個鏡像都從相同的BASE鏡像構建來的,那么宿主機只需要在磁盤上保留1分BASE鏡像,同時內存中也只需要加載一份BASE鏡像,這樣所有的容器都可以使用。另外,鏡像的每一層都是可以共享的。
如何查看鏡像分層?
可以通過docker image inspect來查看鏡像的分層,比如查看剛才下載的redis鏡像:
docker image inspect redis:latest
所有的docker鏡像都起始于一個基礎鏡像層,當進行修改或者增加新的內容時,就會在當前鏡像層之上,創建新的鏡像層。
就好比:修房子,首先有了地基,然后蓋了第一層后并裝修,開始蓋第二層,你發現想要的東西,第一層已經有了,那么就不會重復去修建了,只會修不一樣,新的裝飾。
四、Docker容器
鏡像(Image)和容器(Container)的關系,就像是面向對象程序設計中的類和實例一樣,鏡像是靜態的定義,容器是鏡像運行時的實體。容器可以被創建、啟動、停止、刪除、暫停等 。
Docker | 面向對象 |
---|---|
容器 | 對象 |
鏡像 | 類 |
容器的實質是進程,但與直接在宿主執行的進程是不同,容器進程運行于屬于自己的獨立的命名空間。因此容器可以擁有自己的 root 文件系統、自己的網絡配置、自己的進程空間,甚至自己的用戶 ID 空間。
五、Docker鏡像與容器的形象比喻
請看上一章節:大白話帶你快速安裝Docker,不懂你捶我
原文鏈接:https://blog.csdn.net/shujuelin/article/details/123142292
相關推薦
- 2022-04-12 C#實現六大設計原則之里氏替換原則_C#教程
- 2023-01-20 Python輸入整數進行排序方式_python
- 2022-05-21 詳解C#枚舉中使用Flags特性_C#教程
- 2022-08-16 Kotlin實用語法糖空安全類型轉換及相等性判斷_Android
- 2022-11-30 詳解Golang中字符串的使用_Golang
- 2022-08-01 為MongoDB數據庫注冊windows服務_MongoDB
- 2023-07-09 echart 設置柱狀圖y軸最大刻度
- 2022-10-18 CommonLisp中解析命令行參數示例_其它綜合
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支