用戶(hù)在使用docker的過(guò)程中蓬痒,往往需要能查看容器內(nèi)應(yīng)用產(chǎn)生的數(shù)據(jù),或者需要把容器內(nèi)的數(shù)據(jù)進(jìn)行備份,甚至多個(gè)容器之間進(jìn)行數(shù)據(jù)的共享垄潮,這必然涉及容器的數(shù)據(jù)管理操作。
容器中管理數(shù)據(jù)主要有兩種方式:
- 數(shù)據(jù)卷(Data Volumes)
- 數(shù)據(jù)卷容器(Data Volumes Containers)
本節(jié)將首先介紹如何在容器內(nèi)創(chuàng)建數(shù)據(jù)卷闷盔,并且把本地的目錄或文件掛載到容器內(nèi)的數(shù)據(jù)卷中弯洗。接下來(lái)會(huì)介紹如何使用數(shù)據(jù)卷容器在容器和主機(jī)、容器和容器之間共享數(shù)據(jù)逢勾,并實(shí)現(xiàn)數(shù)據(jù)的備份和恢復(fù)牡整。
數(shù)據(jù)卷
數(shù)據(jù)卷是一個(gè)可供容器使用的特殊目錄,它繞過(guò)文件系統(tǒng)溺拱,可以提供很多有用的特性逃贝。
- 數(shù)據(jù)卷可以在容器之間共享和重用
- 對(duì)數(shù)據(jù)卷的修改會(huì)立馬生效
- 對(duì)數(shù)據(jù)卷的更新不會(huì)影響鏡像
- 卷會(huì)一直存在,直到?jīng)]有容器使用
數(shù)據(jù)卷的使用迫摔,類(lèi)似于Linux下對(duì)目錄或文件進(jìn)行mount操作沐扳。
在容器內(nèi)創(chuàng)建一個(gè)數(shù)據(jù)卷
在用docker run命令的時(shí)候,使用-v標(biāo)記可以在容器內(nèi)創(chuàng)建一個(gè)數(shù)據(jù)卷句占,多次使用-v標(biāo)記可以創(chuàng)建多個(gè)數(shù)據(jù)卷沪摄。
下面使用training/webapp鏡像創(chuàng)建一個(gè)web容器,并創(chuàng)建一個(gè)數(shù)據(jù)卷掛載到容器的/webapp目錄:
docker run -d -P --name web -v /webapp training/webapp python app.py
注意:-P是允許外部訪問(wèn)容器需要暴露的端口
掛載一個(gè)主機(jī)目錄作為數(shù)據(jù)卷
使用-v標(biāo)記也可以指定掛載一個(gè)本地的已有目錄到容器中去作為數(shù)據(jù)卷:
docker run -d -P --name web -v /src/webapp:/opt/webapp training/webapp python app.py
上面的命令加載主機(jī)的/src/webapp目錄到容器的/opt/webapp目錄。用戶(hù)可以放置一些程序或數(shù)據(jù)到本地目錄中杨拐,然后在容器內(nèi)運(yùn)行和使用祈餐。
docker掛載數(shù)據(jù)卷的默認(rèn)權(quán)限是讀寫(xiě)(rw),用戶(hù)也可以通過(guò)ro指定為只讀哄陶。
docker run -d -P --name web -v /src/webapp:/opt/webapp:ro training/webapp python app.py
加了:ro之后昼弟,容器內(nèi)掛載的數(shù)據(jù)卷的數(shù)據(jù)就無(wú)法修改了。
掛載一個(gè)本地主機(jī)文件作為數(shù)據(jù)卷
-v標(biāo)記也可以從主機(jī)掛載單個(gè)文件到容器中作為數(shù)據(jù)卷:
docker run --rm -it -v ~/.bash_history: ~/.bash_history ubuntu /bin/bash
這樣就可以記錄在容器輸入過(guò)的命令歷史了奕筐。(不推薦此種方式舱痘,建議直接掛載文件所在的目錄)
數(shù)據(jù)卷容器
如果用戶(hù)需要在容器之間共享一些持續(xù)更新的數(shù)據(jù),最簡(jiǎn)單的方式就是數(shù)據(jù)卷容器离赫。數(shù)據(jù)卷容器其實(shí)就是一個(gè)普通的容器芭逝,專(zhuān)門(mén)用它提供數(shù)據(jù)卷供其他容器掛載,使用方式如下:
首先創(chuàng)建一個(gè)數(shù)據(jù)卷容器dbdata渊胸,并在其中創(chuàng)建一個(gè)數(shù)據(jù)卷掛載到/dbdata:
docker run -it -v /dbdata --name dbdata ubuntu
然后旬盯,可以在其他容器中使用--volumes-from來(lái)掛載dbdata容器中的數(shù)據(jù)卷,例如創(chuàng)建db1和db2兩個(gè)容器翎猛,并從dbdata容器掛載數(shù)據(jù)卷:
docker run -it --volumes-from dbdata --name db1 ubuntu
docker run -it --volumes-from dbdata --name db2 ubuntu
此時(shí)胖翰,容器db1和db2都掛載同一個(gè)數(shù)據(jù)卷到相同的/dbdata目錄。三個(gè)容器任何一方在該目錄下的寫(xiě)入切厘,其他容器都可以看得到萨咳。
而且還可以使用--volumes-from參數(shù)從多個(gè)容器掛載多個(gè)數(shù)據(jù)卷,還可以從其他已經(jīng)掛載了容器卷的容器來(lái)掛載數(shù)據(jù)卷:
docker run -d --name db3 --volumes-from db1 training/postgres
注意:使用--volumes-from參數(shù)所掛載數(shù)據(jù)卷的容器自身并不需要保持在運(yùn)行狀態(tài)
如果刪除了掛載的容器(包括dbdata疫稿、db1培他、db2),數(shù)據(jù)卷并不會(huì)被自動(dòng)刪除遗座。如果要?jiǎng)h除一個(gè)數(shù)據(jù)卷舀凛,必須在刪除最后一個(gè)還掛載著它的容器時(shí)顯式地使用docker rm -v命令來(lái)指定同時(shí)刪除關(guān)聯(lián)的容器。
利用數(shù)據(jù)卷容器遷移數(shù)據(jù)
可以利用數(shù)據(jù)卷容器對(duì)其中的數(shù)據(jù)卷進(jìn)行備份途蒋、恢復(fù)猛遍,以實(shí)現(xiàn)數(shù)據(jù)的遷移。
備份
使用下面的命令來(lái)備份dbdata數(shù)據(jù)卷容器內(nèi)的數(shù)據(jù)卷:
docker run --volumes-from dbdata -v ${pwd}:/backup --name worker ubuntu tar cvf /backup/backup.tar /dbdata
首先利用ubuntu鏡像創(chuàng)建了一個(gè)容器worker号坡,使用--volumes-from dbdata參數(shù)來(lái)讓worker容器掛載dbdata容器的數(shù)據(jù)卷(即dbdata數(shù)據(jù)卷)懊烤;使用-v ${pwd}:/backup參數(shù)來(lái)掛載本地的當(dāng)前目錄到worker容器的/backup目錄。
worker容器啟動(dòng)后筋帖,使用了tar cvf /backup/backup.tar /dbdata命令來(lái)將/dbdata下內(nèi)容備份為容器內(nèi)的/backup/backup.tar奸晴,即宿主主機(jī)當(dāng)前目錄下的backup.tar。
恢復(fù)
如果要恢復(fù)數(shù)據(jù)到一個(gè)容器日麸,首先創(chuàng)建一個(gè)帶有數(shù)據(jù)卷的容器dbdata2
docker run -v /dbdata --name dbdata2 ubuntu /bin/bash
然后創(chuàng)建另一個(gè)新的容器寄啼,掛載dbdata2的容器逮光,并使用untar解壓備份文件到所掛載的容器卷中即可:
docker run --volumes-from dbdata2 -v ${pwd}:/backup busybox tar xvf /backup/backup.tar