Docker之docker run参数覆盖Dockerfile中CMD命令以及CMD与ENTRYPOINT的区别

 

一:docker run参数覆盖Dockerfile中CMD命令情况

首先要明确docker run指令中的参数是什么?

答案:docker run 指令(-it/-p/-d等) 镜像名 参数(/bin/sh、bash等),即docker run命令中,镜像名后面的 都是给容器执行的命令参数

举例说明docker run参数覆盖Dockerfile中CMD命令的情况:

首先基于镜像创建一个新的镜像,Dockerfile书写如下:

按道理说:用docker run执行使用镜像创建容器时,屏幕上应该会输出:Hello World。

接下来创建镜像:

镜像创建,承建了一个名为test,版本号(tag)为1的镜像

 

然后我们分别用docker run -it test:1 和 docker run -it test:1  bash  两个指令来使用镜像创建启动容器,注意观察他们的区别

一:docker run -it test:1

这里成功输出了Hello World

二:docker run -it test:1  bash

发现并没有输出Hello World, 而是执行了 bash指令给我们提供了交互界面,因此 是docker run -it test  bash中的bash命令参数覆盖了CMD。

关于CMD和ENTRYPOINT还有一点需要特别注意的是:如果一个Dockerfile中有多个CMD或ENTRYPOINT,只有最后一个会生效,前面其他的都会被覆盖

二:CMD与ENTRYPOINT区别

区别一:ENTRYPOINT不会被docker run 中的参数命令覆盖

区别二:如果在Dockerfile中CMD与ENTRYPOINT都存在,则CMD中的指令将会被作为ENTRYPOINT中的参数,即这时候CMD里的指令会失去它的作用,只是作为一个参数(可以理解为就是字符串)了,而且这个时候docker run后的参数指令依然会覆盖CMD,但是也会失去他本身的作用,仅仅是作为参数(字符串)提供给ENTRYPOINT中的命令

例如:一个Dockerfile文件内容如下:

FROM xxx(初始镜像名)

ENTRYPOINT ["echo"]

CMD ["Hello World"]

使用该Dockerfile创建新镜像后,执行 docker run -it 镜像名,就应该会输出:Hello World(没有实际操作验证)

接下来留一个小思考,如果使用该Dockerfile创建完成新镜像之后,执行docker run -it 镜像名 bash指令创建执行容器,CMD ["Hello World"] 这一条会不会被覆盖,会不会显示Hello World?

 

依然会覆盖CMD,因此不会输出Hello World 而是会输出bash,而且 现在的bash已经不是给我们提供交互界面的指令了,而是ENTRYPOINT中 echo的一个参数了,因此也无法给我们提供交互界面

 

从 docker ps -a就可以看出,使用镜像创建运行容器时执行的指令不是bash而是“echo bash“

 

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
# docker搭建lnmp环境 <!-- TOC --> - [docker搭建lnmp环境](#docker搭建lnmp环境) - [一、Dockerfile定制镜像](#一dockerfile定制镜像) - [二、docker-compose](#二docker-compose) - [三、docker-compose编排lnmp环境](#三docker-compose编排lnmp环境) - [1、mysql](#1mysql) - [2、redis](#2redis) - [3、mongo](#3mongo) - [4、nginx](#4nginx) - [5、php](#5php) - [6、完整版](#6完整版) - [四、参考](#四参考) <!-- /TOC --> 有收获的话请**加颗小星星**,没有收获的话可以 **反对** **没有帮助** **举报**三连 ## 一、Dockerfile定制镜像 ```bash # FROM 指定基础镜像 FROM 镜像 FROM php:7.2-fpm # RUN 执行 RUN or RUN ["可执行文件", "参数1", "参数2"] RUN echo 'Hello, Docker!' > /usr/share/nginx/html/index.html RUN ["php", "-S", "0.0.0.0:8080"] # COPY 复制文件 COPY ... COPY swoole-4.2.10.tgz /home COPY nginx.conf /etc/nginx/nginx.conf # ADD 复制文件或目录,如果是.tgz,会被解压缩 ADD ... ADD nginx.conf /etc/nginx/nginx.conf # CMD 容器启动 CMD echo $HOME => CMD [ "/bin/sh", "-c", "echo $HOME" ] CMD [ "redis-server", "/usr/local/etc/redis/redis.conf" ] # ENTRYPOINT 入口点 ENTRYPOINT ["docker-entrypoint.sh"] 存在 ENTRYPOINT 后,CMD 的内容将会作为参数传给 ENTRYPOINT # ENV 环境变量 ENV ENV MYSQL_ROOT_PASSWORD root # ARG与ENV差不多 ARG 所设置的构建环境的环境变量,在将来容器运行时是不会存在这些环境变量的 ENV MYSQL_ROOT_PASSWORD root # VOLUME 匿名卷 VOLUME ["", ""...] VOLUME ["/data"] # EXPOSE 暴露端口 EXPOSE [...] EXPOSE 80 443 # WOEKDIR 指定工作目录,进入容器后的落地目录 WORKDIR WORKDIR /var/www # USER 指定当前用户 USER USER root ``` ## 二、docker-compose - 服务 (service):一个应用的容器,实际上可以包括若干运行相同镜像的容器实例。 - 项目 (project):由一组关联的应用容器组成的一个完整业务单元,在 docker-compose.yml 文件定义。 ## 三、docker-compose编排lnmp环境 ### 1、mysql 这里我们使用了mysql5.5版本,没其它用意,相比5.7以上版本,占内存和硬盘最小的一个版本 我们准备了一个`my.cnf`作为额外配置,这里我修改了数据库的时区 ```cnf [mysqld] default-time-zone = '+8:00' ``` ```Dockerfile FROM mysql:5.5 COPY my.cnf /etc/mysql/conf.d EXPOSE 3306 ``` ### 2、redis 我们使用准备的配置文件`redis.conf`覆盖容器默认启动的配置文件,修改了`ip绑定`和`密码` ```conf bind 0.0.0.0 requirepass root ``` ```Dockerfile FROM redis:latest COPY redis.conf /usr/local/etc/redis/redis.conf CMD [ "redis-server", "/usr/local/etc/redis/redis.conf" ] EXPOSE 6379 ``` ### 3、mongo mongodb我们没有特殊处理 ```Dockerfile FROM mongo:latest EXPOSE 27017 ``` ### 4、nginx 我们准备了一份`nginx.conf`和虚拟目录`conf.d`,为了以后可以动态的配置网站的代理和负载均衡 还有一个日志目录,放在外层`logs`目录里面,记录nginx的访问日志 特别注意的是`fastcgi_pass php:9000;`而不是`fastcgi_pass 127.0.0.1:9000;`,目前自己也没明白 ```Dockerfile FROM nginx:alpine COPY nginx.conf /etc/nginx/nginx.conf EXPOSE 80 ``` ### 5、php php算是这里面最难搞定的,因为我们需要额外的添加php扩展,虽然php的docker官方提供了`docker-php-ext-configure`, `docker-php-ext-install`, `docker-php-ext-enable`,还是有些扩展需要通过`手动编译`或者`pecl`安装 由于pecl官网下载慢,我们事先下载好了几个需要的扩展 php-fpm用的是debian的linux系统,下载也很慢,我们备用了阿里云的镜像`sources.list` 我们还准备了php的默认配置`php.ini`和`opcache.ini` 比如swoole扩展安装,记得安装包用完后清理,还有得用`COPY`命令,`ADD`会解压缩 ```Dockerfile # swoole COPY swoole-4.2.10.tgz /home RUN pecl install /home/swoole-4.2.10.tgz && \ docker-php-ext-enable swoole && \ rm /home/swoole-4.2.10.tgz ``` ### 6、完整版 ```yml version: '3' networks: frontend: driver: bridge backend: driver: bridge volumes: mysql: driver: local mongo: driver: local redis: driver: local services: php: build: ./php volumes: - ${WORKER_DIR}:/var/www ports: - 9100:9000 depends_on: - mysql - redis - mongo networks: - backend nginx: build: ./nginx volumes: - ${WORKER_DIR}:/var/www - ./logs/nginx:/var/log/nginx - ./nginx/conf.d:/etc/nginx/conf.d ports: - 8000:80 depends_on: - php networks: - frontend - backend mysql: build: ./mysql environment: - MYSQL_ROOT_PASSWORD=root volumes: - ${DATA_PATH}/mysql:/var/lib/mysql ports: - 3310:3306 networks: - backend mongo: build: ./mongo environment: - MONGO_INITDB_ROOT_USERNAME=root - MONGO_INITDB_ROOT_PASSWORD=root ports: - 27010:27017 volumes: - ${DATA_PATH}/mongo:/data/db networks: - backend redis: build: ./redis volumes: - ${DATA_PATH}/redis:/data ports: - 6310:6379 networks: - backend ```
Docker 是一种轻量级的容器化技术,可以将应用程序及其依赖项打包在一个可移植的容器,这种技术在现代软件开发和运行环境越来越流行。当需要在同一个容器同时运行多个服务或应用程序时,我们可以使用 DockerCMDENTRYPOINT 命令来实现。 CMD 命令用于指定容器启动时默认运行的命令,并且可以被覆盖。在 CMD 命令,我们可以指定多条命令,以分号隔开。例如: ``` CMD ["command1", "command2"] ``` 当容器启动时,会自动执行 command1 和 command2。 ENTRYPOINT 命令也类似于 CMD 命令,用于指定容器启动时要执行的命令。不同的是,ENTRYPOINT 命令可以在容器启动时不能被覆盖。如果用户在使用 docker run 命令启动容器时,指定了其他命令,这些命令会被传递给 ENTRYPOINT 命令。例如: ``` ENTRYPOINT ["command1"] CMD ["command2"] ``` 当容器启动时,会自动执行 command1,并且将命令参数传递给它。如果用户使用 docker run 命令传递了其他命令参数,那么这些参数会被传递给 CMD 命令执行。 使用这些命令可以很好地控制 Docker 容器同时运行的多个服务或应用程序。例如,我们可以指定同时运行一个 Web 服务器和一个数据库服务器的 Docker 容器,并使用 CMDENTRYPOINT 命令来启动这两个服务。要注意的是,这些服务需要在同一个容器运行,会占用更多的计算资源和内存,因此需要根据实际情况进行优化和调整。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

吃苹果的程序猿

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

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

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

打赏作者

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

抵扣说明:

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

余额充值