当前位置: 首页 > news >正文

为什么不建议在 Docker 中跑 MySQL?

前言

今天我们来聊聊一个很有趣的话题:为什么我不建议在Docker中运行MySQL数据库?

有些小伙伴在工作中可能为了部署方便,习惯将所有组件都容器化,但数据库真的适合放在容器里吗?

今天就专门跟大家一起聊聊这个话题,希望对你会有所帮助。

一、容器化与数据库:天生的矛盾?

让我们先思考一个基本问题:容器设计的初衷是什么?

Docker官网明确说明:"容器是进程的隔离环境,适合运行无状态服务"。

而MySQL正是一个典型的有状态服务

image

从这张图可以清晰看出,MySQL作为有状态服务,在容器化环境中面临着独特的挑战。

二、性能问题:I/O瓶颈无法避免

有些小伙伴在工作中可能遇到过MySQL在Docker中性能下降的问题,这其实不是偶然现象。

2.1 存储I/O性能损耗

Docker的存储驱动层会增加额外的I/O开销。我们来看一个简单的性能测试对比:

# 测试原生Linux磁盘写入速度
dd if=/dev/zero of=test.bin bs=1G count=1 oflag=direct# 测试Docker容器内磁盘写入速度
docker run --rm -it ubuntu dd if=/dev/zero of=test.bin bs=1G count=1 oflag=direct

在实际测试中,Docker内部的I/O性能通常比原生系统低10%-20%。

对于MySQL这种I/O密集型的应用,这种性能损耗是致命的。

2.2 网络性能开销

虽然Docker的网络性能已经大幅改善,但仍然存在额外开销:

image

每条网络请求在Docker中都需要经过额外的网络栈处理,增加了延迟和CPU开销。

三、数据持久化:容器与数据的生命周期管理

数据丢失风险是Docker中运行MySQL最大的痛点。

3.1 数据卷的陷阱

很多教程会告诉你使用Volume来持久化数据:

docker run -d \--name mysql \-v mysql_data:/var/lib/mysql \-e MYSQL_ROOT_PASSWORD=password \mysql:8.0

但这并不能完全解决问题。考虑以下场景:

  1. 容器意外删除docker rm -f mysql 然后数据卷变成孤儿卷
  2. 备份恢复复杂:需要同时备份容器配置和数据卷
  3. 迁移困难:数据卷在不同主机间的迁移复杂

3.2 数据一致性挑战

MySQL的写操作需要保证数据安全落盘,但在容器环境中:

// 模拟MySQL写操作流程
public class MySQLWriteProcess {public void writeData(Transaction transaction) {// 1. 写入redo logwriteRedoLog(transaction);// 2. 刷新到磁盘flushToDisk(); // 这里受容器I影响// 3. 确认提交confirmCommit();}// 容器崩溃可能导致这一步失败private void flushToDisk() {// 调用系统fsync()// Docker存储驱动增加额外层System.callFsync();}
}

容器崩溃可能导致数据没有完全持久化到物理磁盘。

四、资源管理:无法精确控制

4.1 内存管理问题

MySQL的性能高度依赖正确的内存配置,但Docker的内存限制可能导致问题:

# 限制容器内存为2G
docker run -d --memory=2g --memory-swap=2g mysql

这种情况下,MySQL可能因为内存不足而频繁使用swap,导致性能急剧下降。

4.2 CPU资源竞争

在容器环境中,CPU资源的分配和隔离不如物理机稳定:

image

当宿主机资源紧张时,容器间的CPU竞争会导致MySQL性能不稳定。

五、高可用与故障恢复:复杂度的指数级增长

有些小伙伴在设计系统时,往往低估了数据库高可用的复杂度。

5.1 复制与集群的挑战

在Docker中部署MySQL集群需要解决很多额外问题:

# docker-compose.yml 部分配置
version: '3.8'
services:mysql-master:image: mysql:8.0networks:- mysql-clusterenvironment:- MYSQL_REPLICATION_MODE=master- MYSQL_REPLICATION_USER=repl- MYSQL_REPLICATION_PASSWORD=passwordmysql-slave:image: mysql:8.0networks:- mysql-clusterenvironment:- MYSQL_REPLICATION_MODE=slave- MYSQL_REPLICATION_MASTER=mysql-master

这种配置面临的问题:

  1. 网络延迟:容器间网络通信增加复制延迟
  2. 服务发现:容器IP变化导致复制配置失效
  3. 脑裂风险:容器调度可能导致集群脑裂

5.2 备份恢复的复杂性

在容器环境中实现可靠的备份策略更加复杂:

image

六、安全性与隔离性:隐藏的风险

6.1 安全隔离不足

容器提供的隔离性不如虚拟机,MySQL数据库可能面临安全风险:

  1. 内核共享:所有容器共享宿主机的内核,存在漏洞扩散风险
  2. 资源泄露:通过/proc或/sys可能泄露其他容器信息
  3. 特权升级:配置不当可能导致容器逃逸

6.2 网络安全隐患

Docker的网络模型增加了攻击面:

# 错误的网络配置示例
docker run -d \--network=host \  # 共享主机网络命名空间-p 3306:3306 \mysql

这种配置虽然性能好,但严重降低了安全性。

七、监控与诊断:可见性降低

7.1 监控挑战

在容器中监控MySQL比在物理机上更复杂:

# 容器内监控MySQL
docker exec mysql sh -c \"mysqladmin -uroot -ppassword status"

这种方法的问题:

  1. 需要进入容器执行命令
  2. 监控指标受容器资源限制影响
  3. 难以区分是MySQL问题还是容器环境问题

7.2 诊断困难

当出现性能问题时,诊断容器内的MySQL更加困难:

image

需要同时排查容器环境和MySQL本身的问题,复杂度大大增加。

八、什么时候可以在Docker中运行MySQL?

虽然我不建议在生产环境这样做,但在某些场景下还是可以的:

8.1 开发测试环境

在开发环境中使用Docker运行MySQL有很多好处:

# docker-compose.dev.yml
version: '3.8'
services:mysql:image: mysql:8.0environment:MYSQL_ROOT_PASSWORD: passwordMYSQL_DATABASE: myappports:- "3306:3306"volumes:- ./data:/var/lib/mysql- ./config:/etc/mysql/conf.d

开发环境的优点:

  • 快速搭建和销毁
  • 环境一致性
  • 易于版本切换

8.2 特定生产场景

在满足以下条件时,可以考虑在生产环境使用Docker运行MySQL:

  1. 数据重要性低:可以接受数据丢失的场景
  2. 资源充足:宿主机资源远远超过MySQL需求
  3. 有专业团队:具备深度容器和MySQL知识的团队
  4. 完善的监控:有全面的监控和告警系统

九、生产环境推荐方案

对于生产环境,我推荐以下部署方案:

9.1 传统物理机部署

image

9.2 Kubernetes StatefulSet方案

如果必须在容器环境运行,建议使用Kubernetes StatefulSet:

apiVersion: apps/v1
kind: StatefulSet
metadata:name: mysql
spec:serviceName: "mysql"replicas: 3selector:matchLabels:app: mysqltemplate:metadata:labels:app: mysqlspec:containers:- name: mysqlimage: mysql:8.0resources:requests:memory: "4Gi"cpu: "2"volumeMounts:- name: mysql-datamountPath: /var/lib/mysqlvolumeClaimTemplates:- metadata:name: mysql-dataspec:accessModes: [ "ReadWriteOnce" ]storageClassName: "ssd"resources:requests:storage: 100Gi

总结

经过上面的分析,我们可以得出以下结论:

  1. 性能损耗:Docker的存储和网络栈带来明显的性能开销,不适合I/O密集型的MySQL。

  2. 数据安全:容器与数据生命周期管理复杂,增加数据丢失风险。

  3. 运维复杂度:监控、诊断、备份恢复等在容器环境中更加复杂。

  4. 资源管理:Docker的资源限制可能影响MySQL性能稳定性。

  5. 安全性:容器隔离性不如虚拟机,增加安全风险。

有些小伙伴可能会说:"但是我就是在Docker中跑MySQL,没遇到什么问题啊!"

确实,在小规模、非核心的业务中,你可能不会立即感受到这些问题。

但随着业务增长,这些潜在问题会逐渐暴露。

我的建议是:在开发测试环境可以大胆使用Docker运行MySQL,但在生产环境尤其是核心业务中,应该慎重考虑传统部署方案或专业的云数据库服务。

数据库是系统的基础,稳定性压倒一切。

不要为了技术的时髦而牺牲系统的可靠性。

毕竟,我们的首要职责是保证系统稳定运行,而不是追求最酷的技术。

最后说一句(求关注,别白嫖我)

如果这篇文章对您有所帮助,或者有所启发的话,帮忙关注一下我的同名公众号:苏三说技术,您的支持是我坚持写作最大的动力。

求一键三连:点赞、转发、在看。

关注公众号:【苏三说技术】,在公众号中回复:进大厂,可以免费获取我最近整理的10万字的面试宝典,好多小伙伴靠这个宝典拿到了多家大厂的offer。

本文收录于我的技术网站:http://www.susan.net.cn

相关文章:

为什么不建议在 Docker 中跑 MySQL?

前言 今天我们来聊聊一个很有趣的话题:为什么我不建议在Docker中运行MySQL数据库? 有些小伙伴在工作中可能为了部署方便,习惯将所有组件都容器化,但数据库真的适合放在容器里吗? 今天就专门跟大家一起聊聊这个话题,希望对你会有所帮助。 一、容器化与数据库:天生的矛盾?…...

reLeetCode 热题 100-1 指针283. 移动零 - MKT

reLeetCode 热题 100-1 指针283. 移动零 class Solution { public:void moveZeroes(vector<int>& nums) {// int cout_=0;// for(int i =0; i<nums.size();i++){// if(nums[i]==0){// cout_++;// }// }// std::cout<< " 0s all …...

解决c# DocX生成的word文档wps打开排版外边距错乱微软office正常问题

public void insertBreak(DocX document, String filename) { DocX tempDocx = DocX.Create(filename); setPageMargin(tempDocx); document.InsertDocument(tempDocx);document.InsertSectionPageBreak(true); }改为public vo…...

The 2025 ICPC Asia East Continent Online Contest (II)

The 2025 ICPC Asia East Continent Online Contest (II)比赛链接 Review 这场非常有参与感哈哈,因为我签到题 C 贪心写了两小时,中间下机若干次让队友过题,写完已经完全不知道队友进度是啥了,后续就当小黄鸭被带飞了哈哈。 Solution C. Jiaxun! 那我确实需要 jiaxun 额额贪…...

工厂方法模式(Factory Method) - 指南

工厂方法模式(Factory Method) - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", monospace !importa…...

拾忆录

████,也即言多██,就是少点██,不容易发生██——来自于多种████的通理 择一███,遇一人██——收集自███,████ 知行合一心学理论——王阳明...

从零搭建RAG应用:跳过LangChain,掌握文本分块、向量检索、指代消解等核心技术实现

RAG(检索增强生成)本质上就是给AI模型外挂一个知识库。平常用ChatGPT只能基于训练数据回答问题,但RAG可以让它查阅你的专有文档——不管是内部报告、技术文档还是业务资料,都能成为AI的参考资源。 很多人第一反应是用LangChain或LlamaIndex这些现成框架,确实能快速搭起来。…...

python高阶技巧

闭包:在函数嵌套前提下,内部函数使用了外部函数的变量,并且外部函数返回了内部函数,我们把这个使用外部函数变量的内部函数叫做闭包 简单闭包: def outer(logo): def inner(msg): print(f"<{logo}>{msg}>{logo}") return inner fn1=outer("黑马程…...

机器视觉之图像处理篇 - 指南

机器视觉之图像处理篇 - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", monospace !important; font-s…...

尝试hikari和jdbctemplate

试着基于jdbctemplate包装一个MysqlHelper类。连接池采用springboot默认的hikari。jdbctemplate提供基本的防注入,它的写法比jdbc好看,jdbc还需要putint,putstr。提供的另一个功能是结果集的转换。写完,测试代码的面貌如下:var sqlhp = new SqlHelper();sqlhp.configAddress…...

配置Nginx根据IP地址进行流量限制以及返回JSON格式数据

要在Nginx中根据IP地址进行流量限制并返回JSON格式数据,你需要结合Nginx的 ngx_http_limit_req_module模块和一些配置技巧。这个模块允许你基于定义的键值,比如IP地址,限制请求的速率。不过在进入细节前,别忘了备份你的Nginx配置文件 划重点:配置透视战斗护甲 (limit_req_…...

回归

最近因为████导致██发生████,长期的██也不是办法,我决定以███████发文。 我的很多比如███,███都发生了██,所以██的██████视█。 重新██...

CSS纯文本渐变动效

创建一个令人印象深刻的CSS文本渐变动效就像是在文字上施展魔法。想象你的文字就像是一幅幻灯片,色彩在背后流转,让每个字母都像是被彩虹绘制过一样。 为了让这种魔法发生,你需要进入CSS的巫术领地。我们将把渐变动效的制作分解为简单步骤,这样即使你不是CSS的大师,也能轻…...

泛微流程共享

第一步: 第二步:打开合同审批数据,点击右键,选择共享。 第三步:共享权限的查看,可见Giada没有查看权限。 第四步:添加权限,依次进入下面的选项。...

MySQL报错:未知系统变量tx_isolation及隔离级别查询

MySQL在其各个版本中进行了诸多变更和优化,包括系统变量、参数命名、功能等方面的调整。在这个情况中,遇到“未知系统变量tx_isolation”这个错误是因为在MySQL 8.0及以后的版本中,系统变量 tx_isolation已经被重命名为 transaction_isolation。 如果你像老朋友一样寻找 tx_…...

Redssion

1.使用 // 设置锁定资源名称 RLock disLock = redissonClient.getLock("DISLOCK"); //尝试获取分布式锁 boolean isLock= disLock.tryLock(500, 15000, TimeUnit.MILLISECONDS); if (isLock) {try {//TODO if get lock success, do something;Thread.sleep(15000);} …...

if __name__ == __main__:

if __name__ == "__main__": 是 Python 中的一个标准代码块,用于检查一个脚本是否是直接运行的。 工作原理 当一个 Python 脚本被解释器执行时,它会自动定义一些特殊变量。其中一个就是 __name__。如果脚本是直接运行的,Python 会将 __name__ 变量的值设置为 &quo…...

提升系统可靠性:Air8000多串口硬件设计的黄金法则

串口通信的可靠性直接影响工业系统的连续性。Air8000以多串口工业级连接力赋能设备互联,而硬件设计则是其可靠性的根基。总结黄金法则,从信号隔离、阻抗匹配到热设计,全方位保障串口通信的稳定性与安全性。 本文主要从硬件设计的角度,分享串口设计中的一些关键注意点,软件…...

20250915笔记

svn 版本控制工具 一、svn介绍 二、svn安装 1、下载客户端和服务端 安装流程: (1)先安装服务端 (2)在服务端创建仓库 (3)新建用户,新建用户组 (4)设置权限,服务端安装成功 (5)安装客户端(也叫小乌龟) (6)安装桌面右键连接仓库 (7)输入账号和密码 (8)连接…...

enumerate函数

enumerate() 是 Python 中一个非常实用的内置函数,它用于在遍历一个可迭代对象(如列表、元组、字符串等)的同时,获取每个元素的索引和值。 为什么需要 enumerate()? 在没有 enumerate() 之前,如果你想同时获取索引和值,通常需要手动维护一个计数器: fruits = [apple, b…...

2025国内 HR SaaS 竞争格局:易路以AI深度融合引领行业转型

在中国企业数智化转型的浪潮中,HR SaaS 市场正经历从基础数字化向智能协同的关键跃迁。随着全球化布局与本地化合规要求的双重驱动,中大型企业对人力资源管理系统的需求已从单一模块效率提升转向全流程智能协同与全球合规管理。截至 2025 年,中国 HR SaaS 市场规模已突破 30…...

HyperWorks许可激活

在工程项目中,高效的软件工具是成功的关键。而一个顺畅的许可激活流程,则是确保这些工具能够迅速投入使用的重要环节。HyperWorks,作为一款领先的工程仿真软件,以其快速、简便的许可激活流程,为用户提供了卓越的使用体验。 一、一键激活,轻松上手 HyperWorks的许可激活流…...

f-string用法

在 Python 3.6 及更高版本中,在字符串前加上一个 f,表示这是一个 f-string(格式化字符串字面量)。 f-string 的主要作用是让你在字符串中嵌入 Python 表达式,使得格式化字符串变得非常简洁和直观。 f-string 的基本用法 你只需要在字符串开头加上 f,然后在字符串内部用花…...

OpenStack Nova instance 常见操作

1. 启动实例(start) 场景:启动处于 SHUTOFF 状态的实例 源码路径:API 层:nova/compute/api.py → start() RPC 层:nova/compute/rpcapi.py → start_instance() 执行层:nova/compute/manager.py → start_instance() 驱动层:nova/virt/libvirt/driver.py → power_on()…...

libdpi.dll libdatareport.dll libdash_plugin.dll libcurl-x86.dll libcurl-x64.dll libcurl_x64.dll - 指南

libdpi.dll libdatareport.dll libdash_plugin.dll libcurl-x86.dll libcurl-x64.dll libcurl_x64.dll - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas"…...

理解 Kubernetes CSI

关于 Kubernetes CSI,现在的资料已经不少,但我仍希望有一篇文档能让人轻松但不失准确地理解 CSI。 本文不涉及代码分析和详细设计。但需要如下基础:会使用至少一种容器,Docker,containerd,Kata 之类的都可以。 protobuf 和 gRPC:会用并有少量的开发经验,会用某种语言(…...

9.15

开学...

常用数学定理公式

二项式定理 \[(x + y)^n = \sum_{k=0}^n \binom{n}{k} x^{n-k} y^k \]...

线性规划

线性规划是求一个线性函数在满足一组线性等式或不等式方程条件下极值的一类数学问题的统称。要求目标函数和约束方程必须是线性函数。隐含了如下假定:  比例性假定:决策变量的变化与资源消耗成比例;  可加性假定:每个决策变量的影响独立于其他变量;  连续性假定:决…...

伪代码学习总结

伪代码学习总结 1. 什么是伪代码伪代码(Pseudocode) 是一种 算法描述语言。它既不同于自然语言(太模糊),也不是某种具体编程语言(太依赖语法)。使用伪代码的目的:使被描述的算法可以容易地翻译成任何一种编程语言(如 Pascal, C, Java, Python 等); 要求 结构清晰、可…...

20号胶 2511

...

9.13linux系统命令

Linux2系统命令 1、df 查看磁盘使用情况 (1)df 查看磁盘使用情况 Filesystem:代表该文件系统时哪个分区,所以列出的是设备名称。 1K-blocks:说明下面的数字单位是1KB,可利用-h或-m来改变单位大小,也可以用-B来设置。 Used:已经使用的空间大小。Available:剩余的空间…...

9.15 svn git

svn版本控制工具 一、svn介绍 SVN是subversion的简称 是一个开放源代码的版本控制系统,通过采用分支管理系统的高 效管理,简而言之就是用于多个人共同开发同一个项目,实现共 享资源,实现最终集中式的管理。 SVN的作用:在项目组当中对需求规格说明书、测试用例、产品 说明书…...

PVC2601

...

利用RabbitMQ与Redis实现消息的延迟传递的策略

RabbitMQ:时间控的快递中心 让我们从RabbitMQ开始,RabbitMQ拥有一个叫做“Dead Letter Exchanges”(DLX)的机制,这个机制基本上就是一个失物招领处。当一个消息未能及时被送达,RabbitMQ会将它转移至DLX。这个DLX与一个或多个队列关联,这样就能处理这些失去方向的消息了。…...

python 按excel的经纬度提取对应栅格tif文件的数值

栅格文件 批量处理代码:# -*- coding:utf-8 -*- """ @author: suyue @file: extract_stations_from_excel.py @time: 2025/09/09 @desc: 从Excel读取站点信息并提取所有站点的CTT值 """ import rasterio import numpy as np import pandas as p…...

麒麟

麒麟点击跳转...

实现我的第一个本地文档问答机器人

本地文档问答机器人 下面是一个完整的本地文档问答机器人实现,涵盖了阶段三的所有核心概念:文档加载、文本分割、向量存储和检索增强生成(RAG)。 完整代码实现 import os.path from typing import Listfrom huggingface_hub import snapshot_download from langchain.chain…...

17、逻辑回归与分类评估 - 从连续到离散的智能判断 - 教程

17、逻辑回归与分类评估 - 从连续到离散的智能判断 - 教程pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "Consolas", "Monaco", "Courier New", m…...

关于32位单片机使用lwip无法访问(ping)外网,只能与同网段设备进行通信的问题解决

核心问题:MAC地址 理论依据:MAC地址是有相应的规范的MAC地址中,前3字节为组织唯一标识符,后24位由厂家自行定义。也就是说前3字节是有规定的,其中前3字节MAC中的前几位也是有着对应的含义的。MAC地址详细的介绍,大家可以在网上搜得到 出现这个问题的核心就是网关对设备mac地…...

044-WEB攻防-PHP应用SQL盲注布尔回显延时判断报错处理增删改查方式

044-WEB攻防-PHP应用&SQL盲注&布尔回显&延时判断&报错处理&增删改查方式 1.演示案例:➢PHP-MYSQL-SQL操作-增删改查 ➢PHP-MYSQL-注入函数-布尔&报错&延迟 ➢PHP-MYSQL-注入条件-数据回显&错误处理 ➢PHP-MYSQL-CMS案例-插入报错&删除延迟…...

多品牌摄像机视频平台EasyCVR海康大华宇视视频平台统一接入方案

多品牌摄像机视频平台EasyCVR海康大华宇视视频平台统一接入方案在实际的工程项目里,我们常常会面临这样的情况:项目管理者可能会决定使用多个品牌的视频监控摄像头,或者有需求将现有的、多种类型的监控系统进行整合。现在,让我们来探讨一下如何实现不同品牌摄像头的连接和使…...

离散数学课堂习题及课后习题 - PPX

课上的习题,不完整,有空再更新第二章 抽屉原理 Background: 简单形式: 把(n+1)个物体放入n个盒子,必有一个盒子中装了两个物体。其实这个也是狄利克雷描述的一个特殊的表述(如果对于一个映射$ X\to Y $ ,如果\(|X|>|Y|\),则\(f\)不可能是单射,也就是会有\(f(x_1)=f(x…...

玻璃2601

前期五浪下跌走完了 开启反弹...

GoFrame框架查询数据表时对字段取别名

两种方式,基于模型的Fields方法dao.User.Where("id",1).Fields("name as `nickname`","id as `uid`").All()基于结构体orm标签的映射关系type UserData struct {Uid int `json:"id" orm:"id"`Nickname string `json:"…...

ubuntu安装mysql矩阵

安装mysql 5.7版本ubuntu版本    mysql版本    xtrabackup版本    ldd显示GLIBC版本    依赖库处理方式18.04       5.7.42      2.4.28        2.27          不需要处理20.04      5.7.42      2.4.28        …...

二十、DevOps落地:Jenkins基础入门(一)

二十、DevOps落地:Jenkins基础入门(一) 目录二十、DevOps落地:Jenkins基础入门(一)1、DevOps初识1.1 什么是DevOps1.2 DevOps相关工具链1.3 什么是CICD?1.4 持续集成CI介绍1.5 持续交付和持续部署CD介绍1.6 什么是Pipeline(流水线)?1.7 Pipeline编排任务的优势1.8 Pi…...

ubuntu 22.04安装mysql5.7

环境Os:ubuntu 22.04 desktop桌面版mysql:mysql-5.7.42-linux-glibc2.12查看操作系统信息root@db:~# ldd --version ldd (Ubuntu GLIBC 2.35-0ubuntu3) 2.35 Copyright (C) 2022 Free Software Foundation, Inc. This is free software; see the source for copying conditions…...

Docker如何获取镜像

可以使用 docker pull 命令来从仓库获取所需要的镜像。...

2025 ICPC 网络赛2 E

E. Zero 矩阵快速幂优化 dp。 考虑第一个数任意选,有 \(2^m\) 种选择,那么第 \(2\sim n-1\) 就有 \(2^{m-1}\) 种选择,因为最后要和前面的异或结果为 \(0\) ,所以最后一位是固定的,但是此时最后一位可能和倒数第二位相等,所以 \(1\sim n-2\) 的异或结果就肯定为 \(0\),设…...