三、DockerFile 定制屬於自己的專屬鏡像


前言

上篇文章我們知道了怎么操作鏡像和容器,到基礎都是從已經存在的鏡像開始的,那我們自己怎樣搭建一個鏡像並使用它呢?接下來就讓我們學習使用dockerfile 創建屬於自己的鏡像吧。

dockerfile

在這之前,我們需要知道dockerfile ,因為我們就是通過dockerfile 來創建鏡像的。那dockerfile 是什么呢?
dockerfile 是一個文件,文件里面是我們寫的一條條的指令,然后通過docker build
命令來構建一個鏡像。
現在難就難在這個指令怎么寫,所以接下讓我們一起看看dockfile 指令吧。

dockerfile 指令

讓我們先來了解一些基本的 dockerfile 指令吧

FROM

FROM <image>
FROM <image>:<tag>
#tag是可選的,默認會使用latest版本的基礎鏡像

from 指令是依賴的基礎鏡像,所謂的定制鏡像,是在其他的鏡像上添加一些我們自己東西,定制成我們自己的鏡像。當然我們也可以不依賴任何鏡像,自己從頭開始搭建。那就使用

FROM  scratch

scratch 其實也是一個docker 鏡像,但是這個鏡像比較難特殊,它是一個虛擬鏡像,里面什么都沒有,是一個空白的鏡像,所以如果想不依賴任何鏡像,可以使用from scratch

那現在又有一個問題了,dockfile 文件中可以出現多個From 么?

在docker 17.05 版本之前是不支持出現多個From 的,一個dockefile只能有一個From 指令,且必須放在文件中的第一行。因為作為基礎鏡像使用。在docker17.05 后支持多From 。表示構建的多重階段,不過最終生成的鏡像還是以最后一個From 基礎鏡像為基礎的。

RUN

run 指令 是表示在鏡像構建時運行的指令。
兩種格式:

#shell格式
run <命令>
eg: run apt-get update

#exec 格式
run ["可執行文件",“參數1”,“參數2”...]

COPY

復制文件的指令

copy 源路徑  目標路徑
#支持通配符
eg:copy hom?.txt /mydir/

ADD

add 是更高級的復制。copy 有的功能它都用,它還能訪問網絡資源,源路徑可以是一個URL。源路徑文件也可以是一個壓縮文件,可以直接解壓。
所以如果想要直接復制一個壓縮包進去的話,就要使用COPY 而不能只用ADD了。
官方建議是能使用COPY 的就使用COPY ,因為COPY 命令語義比較明確就是復制文件,並且ADD 指令會使得鏡像構建緩存失效,使得鏡像構建比較緩慢。

CMD

cmd 指令是表示在運行容器時執行的指令。

#shell 格式
cmd  <命令>
eg:cmd echo $HOME

#exec 格式
cmd ["可執行文件",“參數1”,“參數2”]
eg: cmd ["sh","-c","echo $HOME"]

ENTRYPOINT

entrypoint 入口點

entrypoint <命令>

entrypoint ["可執行文件","參數1","參數2"]

entryPoint 指令和 cmd 指令功能類似,不過entrypoint 可以讓鏡像變成像命令一樣使用,可以做應用運行前的准備工作。這個具體的后面講。

ENV

env 是設置環境變量的指令,

env MY_VERSION 1.0.0

ARG

arg 用於構建時傳遞的參數

arg <參數名>[=<默認值>]

eg: 
arg version
arg myversion=1.0.0

VOLUME

定義匿名卷

volume <路徑>
volume ["<路徑1>",["<路徑2>"...]

eg: volume  /etc/docker/log

EXPOSE

申明端口

expose <端口1>  [<端口2>...]

這里需要注意的是,expose 是申明容器應用端口,但是容器運行是並不一定就是開啟這個端口提供服務。在dockerfile 中寫入端口申明有兩個好處,一是當做鏡像服務的守護端口,方便映射,二是在運行時使用隨機端口映射時,就會映射的expose設置的端口上。

好了,指令當然不止這些,更多的想了解的查看:

https://docs.docker.com/engine/reference/builder/

簡單測試

之前這篇文章寫到一半放下了,因為中間docker出了一點問題,下載鏡像一直提示超時,然后設置了國內加速,才弄好。
上面我們了解了Dockerfile 指令,接下來就讓我們先做一個簡單的測試吧。
我們穿件一個springboot項目。創建一個HelloController 類。

@RestController
public class HelloController {
    @RequestMapping("/")
    public String hello(){
        return "hello docke 我的簡單測試 ";
    }
}

然后打成jar 包。放到我們服務器的文件夾下。
並且在文件下創建Dockerfile文件

vim Dcokerfile

#文件內容
FROM java:8
VOLUME /tmp
ADD hello-1.0.0.jar hello-1.0.0.jar
ENTRYPOINT ["java","-jar","/hello-1.0.0.jar"]

可以看到用到的命令都是我們上面介紹的。java8作為基礎鏡像,/tmp作為數據卷, add 將本地jar包添加到鏡像中,entrypoint 運行我們的jar包。

file

在該目錄下構建鏡像現在,最后面的點別忘記了。

docker build -t helle:v1 .

file
可以看到我們的鏡像分4不構建,也就是構建四個鏡像,因為我們Dockerfile 中有四條指令。前面說了后一條指令是在前一條指令的基礎上構建鏡像的。所以這四個鏡像中前面三個就是中間鏡像了。

我們現在看看我們創建的鏡像。
file

我們接下來啟動鏡像

docker run -d -p 8090:8080 hello:v1 

file
其中 -d 是后台啟動,-p 是映射端口,前面的是我們設置的端口,后面是項目運行的端口。
啟動后我們在瀏覽器上訪問下。

http://192.168.252.53:8090

file

這樣我們通過docker 構建我們springboot 的項目,創建屬於我們自己的鏡像就完成了。

配置docker遠程訪問

我們現在要做的是,直接通過idea打包生成docker鏡像。所以,第一步開啟docker的遠程訪問,我的docker 是安裝到服務器上的。我先在本地檢測一下,服務器上的docker 是否開啟的遠程訪問。

docker -H 192.168.252.53 info

file
說明是沒有開啟docker的遠程服務的。所以進入服務器。
執行如下操作,在docker.service. 文件夾下創建一個http-proxy.conf文件.

sudo mkdir /etc/systemd/system/docker.service.d
sudo vim /etc/systemd/system/docker.service.d/http-proxy.conf

文件內容

[Service]
ExecStart=
ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375 -H unix:///var/run/docker.sock

然后重啟daemon和docker

sudo systemctl daemon-reload
sudo systemctl restart docker

然后我們再 在本地測試一下。說明docker 的遠程訪問已經配置好了。
file

idea配置

我們打開我們的hello 項目,在pom.xml 中增加配置

<properties>
        <java.version>1.8</java.version>
        <docker.image.prefix>quellanan</docker.image.prefix>
</properties>

在build 中增加。

<!-- Docker -->
            <plugin>
                <groupId>com.spotify</groupId>
                <artifactId>docker-maven-plugin</artifactId>
                <version>1.0.0</version>
                <!-- 將插件綁定在某個phase執行 -->
                <executions>
                    <execution>
                        <id>build-image</id>
                        <!-- 用戶只需執行mvn package ,就會自動執行mvn docker:build -->
                        <phase>package</phase>
                        <goals>
                            <goal>build</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                    <!-- 指定生成的鏡像名 -->
                    <imageName>${docker.image.prefix}/${project.artifactId}</imageName>
                    <!-- 指定標簽 -->
                    <imageTags>
                        <imageTag>${project.version}</imageTag>
                    </imageTags>
                    <!-- 指定 Dockerfile 路徑 -->
                    <dockerDirectory>src/main/docker</dockerDirectory>
                    <!-- 指定遠程 docker api地址 -->
                    <dockerHost>http://192.168.252.53:2375</dockerHost>
                    <resources>
                        <resource>
                            <targetPath>/</targetPath>
                            <!-- jar包所在的路徑此處配置的對應target目錄 -->
                            <directory>${project.build.directory}</directory>
                            <!-- 需要包含的jar包,這里對應的是Dockerfile中添加的文件名 -->
                            <include>${project.build.finalName}.jar</include>
                        </resource>
                    </resources>
                </configuration>
            </plugin>

在src/main/docker 中創建Dockerfile 文件,文件內容上面Dockerfile 內容一樣

FROM java:8
VOLUME /tmp
ADD hello-1.0.0.jar hello-1.0.0.jar
ENTRYPOINT ["java","-jar","/hello-1.0.0.jar"]

file

mvn package

因為我們配置在構建的時候就會進行docker 打包。所以我們知己運行mvn package

file
控制台查看是打包成功的。
我們去服務器上看下,有沒有
file
可以看到已經成功了。

番外

這篇總算寫完啦算是,中間自己親自試驗,踩了一路坑,也算是初步弄好了。以后我們項目不用將jar 包放到服務器上再來創建鏡像了,可以直接在我們項目中打包構建鏡像就想構建jar 包一樣簡單。還是可以的吧。

好了,就說這么多啦

后續加油♡

歡迎大家關注個人公眾號 "程序員愛酸奶"

分享各種學習資料,包含java,linux,大數據等。資料包含視頻文檔以及源碼,同時分享本人及投遞的優質技術博文。

如果大家喜歡記得關注和分享喲❤

file


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM