Docker 学习路线:构建和优化容器镜像

容器镜像和Dockerfile

容器镜像是可执行的软件包,包括运行应用程序所需的所有内容:代码、运行时、系统工具、库和设置。通过构建自定义镜像,您可以在任何支持 Docker 的平台上无缝地部署应用程序及其所有依赖项。

Dockerfile

构建容器镜像的关键组件是 Dockerfile。它本质上是一个包含有关如何组装 Docker 镜像的说明的脚本。Dockerfile 中的每个指令都会在镜像中创建一个新层,从而更容易跟踪更改并最小化镜像大小。以下是 Dockerfile 的简单示例:

# Use an official Python runtime as a parent image
FROM python:3.7-slim# Set the working directory to /app
WORKDIR /app# Copy the current directory contents into the container at /app
COPY . /app# Install any needed packages specified in requirements.txt
RUN pip install --trusted-host pypi.python.org -r requirements.txt# Make port 80 available to the world outside this container
EXPOSE 80# Define environment variable
ENV NAME World# Run app.py when the container launches
CMD ["python", "app.py"]

构建镜像

创建 Dockerfile 后,您可以使用 docker build 命令构建镜像。在包含 Dockerfile 的目录中的终端中执行以下命令:

docker build -t your-image-name .

此命令告诉 Docker 使用当前目录中的 Dockerfile(.)构建镜像,并为其指定名称(-t your-image-name)。

检查镜像和层

成功构建后,您可以使用 docker image 命令检查创建的镜像:

docker image ls

要更仔细地查看镜像的各个层,请使用 docker history 命令:

docker history your-image-name

要查看镜像的层,也可以使用 docker inspect 命令:

docker inspect your-image-name

要删除镜像,请使用 docker image rm 命令:

docker image rm your-image-name

将镜像推送到注册表

构建镜像后,您可以将其推送到容器注册表(例如 Docker Hub、Google Container Registry 等),以轻松分发和部署应用程序。首先使用您的凭据登录注册表:

docker login

然后,使用注册表 URL 标记您的镜像:

docker tag your-image-name username/repository:tag

最后,将已标记的镜像推送到注册表:

docker push username/repository:tag

构建容器镜像是使用 Docker 的重要方面,它使您可以轻松打包和部署应用程序。通过创建具有精确指令的 Dockerfile,您可以轻松地构建和分发各种平台的镜像。

高效的层缓存

在构建容器镜像时,Docker 会缓存新创建的层。这些层可以在构建其他镜像时重复使用,减少构建时间并最小化带宽使用。但是,要充分利用这种缓存机制,您需要了解如何有效地使用层缓存。

Docker 层缓存的工作原理

Docker 为 Dockerfile 中的每个指令(例如 RUNCOPYADD 等)创建一个新层。如果指令自上次构建以来没有更改,Docker 将重用现有的层。

例如,考虑以下 Dockerfile:

FROM node:14
WORKDIR /app
COPY package.json .
RUN npm install
COPY . .

第一次构建镜像时,Docker 将执行每个指令并为每个指令创建一个新层。如果对应用程序进行了一些更改并再次构建镜像,Docker 将检查更改的指令是否影响任何层。如果没有任何层受到更改的影响,Docker 将重用缓存的层。

有效使用层缓存的技巧

  • 最小化 Dockerfile 中的更改: 尽量减少 Dockerfile 中更改的频率,并以使最常更改的行出现在底部的方式构造指令。
  • 构建上下文优化: 使用 .dockerignore 文件从构建上下文中排除不必要的文件,这些文件可能会导致缓存失效。
  • 使用较小的基础镜像: 较小的基础镜像减少拉取基础镜像所需的时间以及需要缓存的层数。
  • 利用 Docker 的 -cachefrom 标志: 如果您正在使用 CI/CD 流水线,则可以指定要用作缓存源的镜像。
  • 组合多个指令: 在某些情况下,组合指令(例如 RUN)可以帮助最小化层数,使缓存更有效。

遵循这些最佳实践,您可以优化层缓存过程并减少 Docker 镜像的构建时间,从而使您的开发和部署过程更加高效。

  • Docker 层缓存

镜像大小和安全性

在构建容器镜像时,了解镜像大小和安全性非常重要。镜像的大小会影响容器的构建和部署速度。较小的镜像可以提高构建速度,并减少下载镜

像时的网络开销。安全性也非常重要,因为容器镜像可能包含漏洞,这可能会对您的应用程序造成风险。

减小镜像大小

  • 使用适当的基础镜像: 选择一个更小、更轻量级的基础镜像,仅包含应用程序所需的必要组件。例如,如果可用,请考虑使用官方镜像的“alpine”变量,因为它通常比较小。
FROM node:14-alpine
  • 在单个“RUN”语句中运行多个命令: 每个“RUN”语句都会在镜像中创建一个新的层,这会增加镜像的大小。使用“&&”将多个命令组合到单个“RUN”语句中,以最小化层数并减小最终镜像大小
RUN apt-get update && \apt-get install -y some-required-package
  • 在同一层中删除不必要的文件: 在镜像构建过程中安装软件包或添加文件时,可以在同一层中删除临时或未使用的文件,以减小最终镜像大小。
RUN apt-get update && \apt-get install -y some-required-package && \apt-get clean && \rm -rf /var/lib/apt/lists/*
  • 使用多阶段构建: 使用多阶段构建创建更小的镜像。多阶段构建允许您在 Dockerfile 中使用多个“FROM”语句。每个“FROM”语句在构建过程中创建一个新的阶段。您可以使用“COPY --from”语句将文件从一个阶段复制到另一个阶段。
FROM node:14-alpine AS build
WORKDIR /app
COPY package*.json ./
RUN npm install
COPY . .
RUN npm run buildFROM node:14-alpine
WORKDIR /app
COPY --from=build /app/dist ./dist
COPY package*.json ./
RUN npm install --production
CMD ["npm", "start"]
  • 使用“.dockerignore”文件: 使用“.dockerignore”文件从构建上下文中排除不必要的文件,这些文件可能会导致缓存失效并增加最终镜像大小。
node_modules
npm-debug.log

提高安全性

  • 保持基础镜像更新: 定期更新您在 Dockerfile 中使用的基础镜像,以确保它们包括最新的安全补丁。
  • 避免使用 root 运行容器: 始终在运行容器时使用非 root 用户,以最小化潜在风险。在运行应用程序之前,创建一个用户并切换到该用户。
RUN addgroup -g 1000 appuser && \adduser -u 1000 -G appuser -D appuser
USER appuser
  • 限制“COPY”或“ADD”指令的范围: 具体说明要复制到容器镜像中的文件或目录。避免使用“COPY . .”,因为它可能会意外包含敏感文件。
COPY package*.json ./
COPY src/ src/
  • 扫描镜像以查找漏洞: 使用 Anchore 或 Clair 等工具扫描您的镜像以查找漏洞,并在部署之前修复它们。

遵循这些最佳实践,您将能够构建更高效和安全的容器镜像,从而提高性能并降低应用程序中漏洞的风险。

最后

为了方便其他设备和平台的小伙伴观看往期文章:

微信公众号搜索:Let us Coding,关注后即可获取最新文章推送

看完如果觉得有帮助,欢迎 点赞、收藏、关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/239875.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python如何学习最为高效?

学习Python最有效的方法因人而异,但以下是一些建议,可以帮助你更高效地学习Python: 1.明确学习目标: 在开始学习之前,明确你的学习目标。你想要学习Python的哪些方面?是想要掌握基础语法、编写代码、做项目…

PyTorch加载数据以及Tensorboard的使用

一、PyTorch加载数据初认识 Dataset:提供一种方式去获取数据及其label 如何获取每一个数据及其label 总共有多少的数据 Dataloader:为后面的网络提供不同的数据形式 数据集 在编译器中导入Dataset from torch.utils.data import Dataset 可以在jupyter中查看Dataset官方文档&…

围绕天津这个城市我们可以做哪些课题选题,供大家参考

博主介绍:黄菊华老师《Vue.js入门与商城开发实战》《微信小程序商城开发》图书作者,CSDN博客专家,在线教育专家,CSDN钻石讲师;专注大学生毕业设计教育和辅导。 所有项目都配有从入门到精通的基础知识视频课程&#xff…

confluence操作手册

文章目录 快捷键插入宏插入代码块 自动为标题编号使用方法: 移动页面到其它目录层级下 快捷键 插入宏 英文输入法下输入{ 插入代码块 输入三个会跳出 点编辑可以调整样式 自动为标题编号 安装插件Numbered Headings 下载链接:https://appfire.…

Hive-分区与分桶详解(超详细)

文章目录 前言一、Hive分区1. 什么是分区2. 分区的优势3. 如何创建分区表4. 如何插入分区数据5. 如何查询分区数据6. 分区因素 二、Hive分桶1. 什么是分桶2. 分桶的优势3. 如何创建分桶表4. 如何插入分桶数据5. 如何查询分桶数据6. 分桶因素7. 分区和分桶的综合应用 总结 前言 …

CCCC第一题

假设你有两只手(这个好像不用假设 - -!),在你的面前有一堆的糖,左手拿了a颗,右手拿了b颗,请问你一共拿了多少颗糖。 输入格式: 每次输入两个整数a和b,代表你左手和右手拿糖的颗数&…

asp.net core自定义授权过滤器

//只有登录的用户可以访问 1.统一返回格式 namespace webapi;/// <summary> /// 统一数据响应格式 /// </summary> public class Results<T> {/// <summary>/// 自定义的响应码&#xff0c;可以和http响应码一致&#xff0c;也可以不一致/// </sum…

微服务 Spring Cloud 10,如何追踪微服务调用?服务治理的常见手段

目录 一、服务追踪的作用1、优化系统瓶颈2、优化链路调用3、故障排查4、性能优化5、生成网络拓扑图4、透明传输数据 二、节点管理1、服务调用失败一般有两类原因造成&#xff1a;2、服务调用失败的解决方式&#xff1a;3、服务调用失败的具体解决方式&#xff1a; 三、负载均衡…

设计模式中的设计原则

开闭原则&#xff1a;扩展新的功能但不改变原有的程序设计。 public class test {public static void main(String[] args) {Car benz new Car("Benz");benz.driver();//我们需要扩展方法benz.driver180();} } class Car{private String name;public Car(String nam…

vue3 H5项目中实现PDF预览

是需要npm i vue-pdf-embed 安装这个插件的&#xff0c;可兼容ios/Android&#xff0c;下面是本人的使用实例 <template><div class"conten_box"><vue-pdf-embed v-if"pdfSource.url" :source"pdfSource" /></div> <…

ansible的控制语句

本章内容主要介绍 playbook 中的控制语句 使用when判断语句block-rescue判断循环语句 一个play中可以包含多个task&#xff0c;如果不想所有的task全部执行&#xff0c;可以设置只有满足某个条件才执行这个task&#xff0c;不满足条件则不执行此task。本章主要讲解when 和 blo…

typeof 和 instanceofJS数据类型(js的问题)

原始类型&#xff08;基本类型&#xff09;Undefined Null Boolean Number String引用类型&#xff08;复杂类型&#xff09;Object 1、typeof检测返回对应数据类型 console.log(typeof 123); // number console.log(typeof true); // boolean console.log(typeof "hell…

Epson爱普生手臂机器人与PC通讯 C#

一、Epson手臂配置 1.安装Epson手臂控制软件 安装Epson手臂控制软体EPSON RC+ 7.0(根据实际需求下载应用),可以去官网下载安装。 2.硬件配置 准备一台PC,用网线连接PC和EPSON手臂控制器。 3.在PC上修改IP地址 EPSON手臂默认IP(192.168.0.1),PC IP改为手臂同一网段…

XML简介 (EXtensible Markup Language)

XML简介 (EXtensible Markup Language) 可扩展标记语言 特点 XML与操作系统、编程语言的开发平台无关实现不同系统之间的数据交换 作用 数据交互配置应用程序和网站Ajax基石 XML标签 XML文档内容由一系列标签元素组成 <元素名 属性名"属性值">元素内容&l…

【模式识别】探秘判别奥秘:Fisher线性判别算法的解密与实战

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《模式之谜 | 数据奇迹解码》⏰诗赋清音&#xff1a;云生高巅梦远游&#xff0c; 星光点缀碧海愁。 山川深邃情难晤&#xff0c; 剑气凌云志自修。 目录 &#x1f30c;1 初识模式识…

如何通过蓝牙串口启动智能物联网?

1、低功耗蓝牙(BLE)介绍 BLE 技术是一种低成本、短距离、可互操作的鲁棒性无线技术&#xff0c;工作在免许可的 2,4 GHZ 工业、科学、医学(Industrial Scientific Medical&#xff0c;ISM)频段。BLE在设计之初便被定位为一种超低功耗(Ultra Low Power&#xff0c;ULP)无线技术&…

C语言实现UCS2、UTF8与GBK2312编码转换

一、引言 在软件开发中&#xff0c;字符编码是一个非常重要的概念。不同的编码方式用于在不同的系统和应用中表示文本数据。UCS2、UTF8和GBK2312是三种常见的字符编码方式。为了实现不同编码间的转换&#xff0c;我们可以使用C语言进行编程&#xff0c;利用已有的库或者手动实…

浅谈Guava Cache的参数使用

CacheLoader 用于数据加载方式比较固定且统一的场景&#xff0c;在缓存容器创建的时候就需要指定此具体的加载逻辑。通常开发中使用时我们需要继承CacheLoader类或写一个匿名实现类实现其load方法和reload方法 load方法 当执行get操作没有命中缓存或者判断缓存已经超出expir…

【Prometheus|报错】Out of bounds

【背景】进入Prometheus地址的9090端口&#xff0c;pushgateway&#xff08;0/1&#xff09;error : out of bounds 【排查分析】 1、out of bounds报错&#xff0c;是由于Prometheus向tsdb存数据出错&#xff0c;与最新存数据的时间序列有问题&#xff0c;有可能当前时间与最…

「微服务模式」七种微服务反模式

什么是微服务 流行语经常为进化的概念提供背景&#xff0c;并且需要一个良好的“标签”来促进对话。微服务是一个新的“标签”&#xff0c;它定义了我个人一直在发现和使用的领域。文章和会议描述了一些事情&#xff0c;我慢慢意识到&#xff0c;过去几年我一直在发展自己的个人…