Docker高级篇之DockerFile解析

7 篇文章 0 订阅
2 篇文章 0 订阅

1. DockerFile是什么?

Dockerfile是用来构建Docker镜像的文本文件,是由一条条构建镜像所需的指令和参数构成的脚本。

概述
在这里插入图片描述
官网:https://docs.docker.com/engine/reference/builder/
构建三步骤

  1. 编写Dockerfile文件
  2. docker build命令构建镜像
  3. docker run依镜像运行容器实例

2. DockerFile解析过程

2.1 Dockerfile内容基础知识

2.1.1 每条保留字指令都【必须为大写字母】且后面要跟随至少一个参数

2.1.2 指令按照从上到下,顺序执行

2.1.3 #表示注释

2.1.4 每条指令都会创建一个新的镜像层并对镜像进行提交

2.2 Docker执行Dockerfile的大致流程

2.2.1 docker从基础镜像运行一个容器

2.2.2 执行一条指令并对容器作出修改

2.2.3 执行类似docker commit的操作提交一个新的镜像层

2.2.4 docker再基于刚提交的镜像运行一个新容器

2.2.5 执行dockerfile中的下一条指令直到所有指令都执行完成

2.3 小总结

从应用软件的角度来看,Dockerfile、Docker镜像与Docker容器分别代表软件的三个不同阶段,

  1. Dockerfile是软件的原材料
  2. Docker镜像是软件的交付品
  3. Docker容器则可以认为是软件镜像的运行态,也即依照镜像运行的容器实例
    Dockerfile面向开发,Docker镜像成为交付标准,Docker容器则涉及部署与运维,三者缺一不可,合力充当Docker体系的基石。

在这里插入图片描述

  1. Dockerfile,需要定义一个Dockerfile,Dockerfile定义了进程需要的一切东西。Dockerfile涉及的内容包括执行代码或者是文件、环境变量、依赖包、运行时环境、动态链接库、操作系统的发行版、服务进程和内核进程(当应用进程需要和系统服务和内核进程打交道,这时需要考虑如何设计namespace的权限控制)等等;

  2. Docker镜像,在用Dockerfile定义一个文件之后,docker build时会产生一个Docker镜像,当运行 Docker镜像时会真正开始提供服务;

  3. Docker容器,容器是直接提供服务的。

3. DockerFile常用保留字指令

参考tomcat8的dockerfile入门:https://github.com/docker-library/tomcat

3.1 FROM

基础镜像,当前新镜像是基于哪个镜像的,指定一个已经存在的镜像作为模板,第一条必须是from

3.2 MAINTAINER

镜像维护者的姓名和邮箱地址

3.3 RUN

容器构建时需要运行的命令

两种格式

  1. shell格式
    在这里插入图片描述
RUN yum -y install vim
  1. exec格式
    在这里插入图片描述

RUN是在 docker build时运行

3.4 EXPOSE

当前容器对外暴露出的端口

3.5 WORKDIR

指定在创建容器后,终端默认登陆的进来工作目录,一个落脚点

3.6 USER

指定该镜像以什么样的用户去执行,如果都不指定,默认是root

3.7 ENV

用来在构建镜像过程中设置环境变量

ENV MY_PATH /usr/mytest

这个环境变量可以在后续的任何RUN指令中使用,这就如同在命令前面指定了环境变量前缀一样;
也可以在其它指令中直接使用这些环境变量,

比如:WORKDIR $MY_PATH

3.8 ADD

将宿主机目录下的文件拷贝进镜像且会自动处理URL和解压tar压缩包

3.8 COPY

类似ADD,拷贝文件和目录到镜像中。 将从构建上下文目录中 <源路径> 的文件/目录复制到新的一层的镜像内的 <目标路径> 位置

COPY src dest

COPY ["src", "dest"]

<src源路径>:源文件或者源目录

<dest目标路径>:容器内的指定路径,该路径不用事先建好,路径不存在的话,会自动创建。

3.9 VOLUME

容器数据卷,用于数据保存和持久化工作

4.0 CMD

指定容器启动后的要干的事情
在这里插入图片描述
注意
Dockerfile 中可以有多个 CMD 指令,但只有最后一个生效,CMD 会被 docker run 之后的参数替换

参考官网Tomcat的dockerfile演示讲解

在这里插入图片描述

官网最后一行命令

在这里插入图片描述
我们演示自己的覆盖操作

它和前面RUN命令的区别

  1. CMD是在docker run 时运行。
  2. RUN是在 docker build时运行。
    在这里插入图片描述
    在这里插入图片描述

4.1 ENTRYPOINT

也是用来指定一个容器启动时要运行的命令

类似于 CMD 指令,但是ENTRYPOINT不会被docker run后面的命令覆盖, 而且这些命令行参数会被当作参数送给 ENTRYPOINT 指令指定的程序

命令格式和案例说明
在这里插入图片描述
ENTRYPOINT可以和CMD一起用,一般是变参才会使用 CMD ,这里的 CMD 等于是在给 ENTRYPOINT 传参。
当指定了ENTRYPOINT后,CMD的含义就发生了变化,不再是直接运行其命令而是将CMD的内容作为参数传递给ENTRYPOINT指令,他两个组合会变成
在这里插入图片描述
案例如下:假设已通过 Dockerfile 构建了 nginx:test 镜像:

在这里插入图片描述

是否传参按照dockerfile编写执行传参运行
Docker命令docker run nginx:testdocker run nginx:test -c /etc/nginx/new.conf
衍生出的实际命令nginx -c /etc/nginx/nginx.confnginx -c /etc/nginx/new.conf

优点:在执行docker run的时候可以指定 ENTRYPOINT 运行所需的参数。

注意:如果 Dockerfile 中如果存在多个 ENTRYPOINT 指令,仅最后一个生效。
在这里插入图片描述

4.2 小总结

在这里插入图片描述

4. 实战案例

自定义镜像mycentosjava8

Centos7镜像具备vim+ifconfig+jdk8
JDK的下载镜像地址
官网下载地址:https://www.oracle.com/java/technologies/downloads/#java8

这个地址也可以:https://mirrors.yangxingzhen.com/jdk/

4.1 新建路径myfile

 midir myfile

4.2 上传下载的jar包

在这里插入图片描述

4.3 同一个目录下编写DockerFile

FROM centos
MAINTAINER zzyy<zzyybs@126.com>
 
ENV MYPATH /usr/local
WORKDIR $MYPATH
 
#安装vim编辑器
RUN yum -y install vim
#安装ifconfig命令查看网络IP
RUN yum -y install net-tools
#安装java8及lib库
RUN yum -y install glibc.i686
RUN mkdir /usr/local/java
#ADD 是相对路径jar,把jdk-8u171-linux-x64.tar.gz添加到容器中,安装包必须要和Dockerfile文件在同一位置
ADD jdk-8u171-linux-x64.tar.gz /usr/local/java/
#配置java环境变量
ENV JAVA_HOME /usr/local/java/jdk1.8.0_171
ENV JRE_HOME $JAVA_HOME/jre
ENV CLASSPATH $JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar:$JRE_HOME/lib:$CLASSPATH
ENV PATH $JAVA_HOME/bin:$PATH
 
EXPOSE 80
 
CMD echo $MYPATH
CMD echo "success--------------ok"
CMD /bin/bash

安装包和Dockerfile必须在同一个位置。

4.3 构建镜像:同一个目录

docker build -t centosjava8:1.5 .

在这里插入图片描述
这是一个漫长的过程

4.4 出现错误

Failed to download metadata for repo 'appstream': Cannot prepare internal mirrorlist: No URLs in mirrorlist

在这里插入图片描述

4.5 问题原因:

在dockerhub里面,centos最新版是centos8,在编写dockersfile的时候

命令拉取镜像就拉取了最新版本,目前应该是centos8,这导致了在追加命令 使用yum源的时候报错

FROM centos

听说 centos8是在21年12月31日停止了源的服务导致,所以我这边只能重新拉取centos7的镜像,即把命令修改了下.

FROM centos:7

在这里插入图片描述
构建镜像,这次构建成功了
在这里插入图片描述

4.6 运行新构建的镜像

 docker run  -it centosjava8:1.5 /bin/bash
 java -version
 ifconfig

在这里插入图片描述

4.7 再体会下UnionFS(联合文件系统)

UnionFS(联合文件系统):Union文件系统(UnionFS)是一种分层、轻量级并且高性能的文件系统,它支持对文件系统的修改作为一次提交来一层层的叠加,同时可以将不同目录挂载到同一个虚拟文件系统下(unite several directories into a single virtual filesystem)。Union 文件系统是 Docker 镜像的基础。镜像可以通过分层来进行继承,基于基础镜像(没有父镜像),可以制作各种具体的应用镜像。
特性:一次同时加载多个文件系统,但从外面看起来,只能看到一个文件系统,联合加载会把各层文件系统叠加起来,这样最终的文件系统会包含所有底层的文件和目录

5. Docker微服务实战

5.1 通过IDEA新建一个普通微服务模块

5.1.1 建Module

docker_boot

5.1.2 改POM

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.5.6</version>
        <relativePath/>
    </parent>

    <groupId>com.atguigu.docker</groupId>
    <artifactId>docker_boot</artifactId>
    <version>0.0.1-SNAPSHOT</version>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
        <junit.version>4.12</junit.version>
        <log4j.version>1.2.17</log4j.version>
        <lombok.version>1.16.18</lombok.version>
        <mysql.version>5.1.47</mysql.version>
        <druid.version>1.1.16</druid.version>
        <mapper.version>4.1.5</mapper.version>
        <mybatis.spring.boot.version>1.3.0</mybatis.spring.boot.version>
    </properties>

    <dependencies>
        <!--SpringBoot通用依赖模块-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>
        <!--test-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <version>3.1.0</version>
            </plugin>
        </plugins>
    </build>

</project>

5.1.3 写YML

server.port=6001

5.1.4 主启动

package com.longxi;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

/**
 * @author LongXi
 * @create 2022-07-12 22:45
 */
@SpringBootApplication
public class DockerBootApplication {
    public static void main(String[] args) {
        SpringApplication.run(DockerBootApplication.class, args);
    }
}

5.1.5 业务类

package com.longxi.controller;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;

import java.util.UUID;

/**
 * @author LongXi
 * @create 2022-07-12 22:52
 */
@RestController
public class OrderController {
    @Value("${server.port}")
    private String port;

    @RequestMapping("/order/docker")
    public String helloDocker() {
        return "hello docker" + "\t" + port + "\t" + UUID.randomUUID().toString();
    }

    @RequestMapping(value = "/order/index", method = RequestMethod.GET)
    public String index() {
        return "服务端口号: " + "\t" + port + "\t" + UUID.randomUUID().toString();
    }
}

5.2 通过dockerfile发布微服务部署到docker容器

5.2.1 IDEA工具里面搞定微服务jar包

在这里插入图片描述
在这里插入图片描述

5.2.2 编写Dockerfile

# 基础镜像使用java
longxi_docker

5.2.3 打包成镜像文件

在这里插入图片描述

 docker build -t longxi_docker:1.6 .

在这里插入图片描述

5.2.4 运行容器

在这里插入图片描述

docker run -d -p 6001:6001 longxi_docker:1.6

在这里插入图片描述

5.2.5 测试一下

在这里插入图片描述
访问成功了

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

东北亚大中华区首席搬砖工具人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值