mysql数据目录结构是怎么样的

66次阅读
没有评论

共计 10377 个字符,预计需要花费 26 分钟才能阅读完成。

mysql 数据目录结构是怎么样的,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面丸趣 TV 小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

mysql 数据目录结构 (转)[@more@] 从概念上讲,大多数关系数据库系统是相似的:它们有一系列数据库组成,每个数据库
包含一系列数据库表,但每个系统有各自组织其管理的数据方式,mysql 也不例外。
缺省地,所有由 mysql 服务器 mysqld 管理的数据存储在一个称为 mysql 数据目录的地方,
所有数据库都存放在哪儿,也包括提供服务器操作信息的状态文件。如果你对一个 mysql 安
装执行管理任务,你应该熟知数据目录的布局及用途。
本文介绍下列专题:
如何确定数据目录的位置。
从概念上讲,大多数关系数据库系统是相似的:它们有一系列数据库组成,每个数据库
包含一系列数据库表,但每个系统有各自组织其管理的数据方式,mysql 也不例外。
缺省地,所有由 mysql 服务器 mysqld 管理的数据存储在一个称为 mysql 数据目录的地方,
所有数据库都存放在哪儿,也包括提供服务器操作信息的状态文件。如果你对一个 mysql 安
装执行管理任务,你应该熟知数据目录的布局及用途。
本文介绍下列专题:
如何确定数据目录的位置。
服务器如何组织并提供对数据库和它管理的表的访问。
在哪里找到由服务器生成的状态文件记忆它们包含什么内容。
如何改变缺省地点或数据目录或单独数据库的组织结构。
1、数据目录的位置
一个缺省数据目录被编译进了服务器,如果你从一个源代码分发安装 mysql,典型的缺
省目录为 /usr/local/var,如果从 RPM 文件安装则为 /var/lib/mysql,如果从一个二进制分
发安装则是 /usr/local/mysql/data。
在你启动服务器,通过使用一个 –datadir=/path/to/dir 选项可以明确指定数据目录位
置。如果你想把数据目录置于其它缺省位置外的某处,这很有用。
作为一名 mysql 管理员,你应该知道你的数据目录在哪里。如果你运行多个服务器,你
应该是到所有数据目录在哪里,但是如果你不知道确切的位置,由多种方法找到它:
使用 mysqladmin variables 从你的服务器直接获得数据目录路径名。查找 datadir 变量
的值,在 Unix 上,其输出类似于:
%mysqladmin variables
+———————-+———————-+| variable_name | Value |+—
——————-+———————-+| back_log | 5 || connect_timeout
| 5 || basedir | /var/local/ || datadir | /usr/local/var/ |….

在 Windows 上,输出可能看上去像这样:
c:mysqladmin variables
+———————-+———————-+| variable_name | Value |+—
——————-+———————-+| back_log | 5 || connect_timeout
| 5 || basedir | c:mysql || datadir | c:mysqldata |….

如果你有多个服务器在运行,它们将在不同的 TCP/IP 端口或套接字上监听,通过提供连
接服务器正在监听的端口或套接字的 –port 或 –socket 选项,你可以轮流获得它们每一个的
数据目录信息:
%msqladmin –port=port_name variables
%mysqladmin –socket=/path/to/socket variables
mysqladmin 命令可运行在任何你能从其连接服务器的主机上,如果你想在一个远程主机
连接服务器,使用一个 –host=host_name 选项:
%mysqladmin –host=host_name variables
在 Windows 上,你可以通过使用 –pipe 强制一个命令管道连接和 –socket=pipe_name 指
定管道名来连接监听一个命令管道的 NT 服务器:
c:mysqladmin –pipe –socket=pipe_name variables
你可以使用 ps 命令查看任何正在运行 mysqld 进程的命令行。
试一下下列命令之一并寻找 –datadir:
%ps axww | grep mysql BSD 风格
%ps -ef | grep mysqld System V 风格
如果你的系统运行多个服务器,ps 命令可能特别有用,因为你能马上发现多个数据目录
位置,缺点是必须在服务器上运行,而且可能没有有用的信息产生,除非在 mysqld 命令行上
明确指定了 –datadir 选项。
如果 mysql 是从一个源代码分发安装的,你可以检查其配置信息确定数据目录位置。例
如,位置可从顶级 Makefile 中获得,但是注意,位置是 Makefile 中的 localstatedir 值,不
是 datadir,而且,如果分发位于一个 NFS 挂载的文件系统并用来为多个主机构建 mysql,配
置信息反映了分发被最新构建的主机,这可能不能提供你感兴趣的主机的数据目录信息。
如果上述方式失败,你可以用 find 寻找数据库文件,下列命令寻找“.frm”文件,它是任
何 mysql 安装的一部分:
% find / -name .frm -print
在下文各例中,用 DATADIR 表示 mysql 数据目录位置。
2、数据目录结构
mysql 数据目录包含了服务器管理的所有数据目录,这些文件被组织成一个树状结构,
通过利用 Unix 或 Windows 文件系统的层次结构直接实现。
每个数据库对应于数据目录下的一个目录。
在一个数据库中的表对应于数据目录下的文件。
数据目录也包含由服务器产生的几个状态文件,如日志文件。这些文件提供了关于服务
器操作的重要信息。对管理特别在出了问题而试图确定问题原因时很有价值。例如,如果某
个特定查询杀死服务器,你可以通过检查日志文件判别捣乱的查询。
2.1 mysql 服务器怎样提供对数据的访问
在数据目录下的一切由一个单独的实体 -mysql 服务器 mysqld 管理,客户程序绝不直接操
作数据。相反,服务器提供数据可访问的切入点,它是客户程序与它们想使用的数据之间的
中介。
当服务器启动时,如果有需要,它打开日志文件,然后通过监听网络连接位数据目录呈
现一个网络接口。要访问数据,客户程序建立对服务器的一个连接,然后以 mysql 查询传输
请求来执行希望的操作。服务器执行每一个操作并将结果发回用户。服务器是多线程的并能
服务多个同时的客户连接。然而,因为修改操作一个执行一个,实际效果是顺序化请求,以
使两个客户决不能在同一时刻改变同一记录。
在正常的情况下,让服务器作为数据库访问的唯一仲裁者提供了避免可从同时访问数据
库表的多个进程的破坏的保证。管理员应该知道有时服务器没有对数据目录的独裁控制。
当你在一个单个数据目录上运行多个服务器。一般倪云新一个服务器管理主机上的所有
数据库,但是有可能运行多个服务器。如果这完成提供对多个独立数据目录的访问,没有相
互影响的问题,但哟也能启动多个服务器并指向同一个目录。一般地,这不是一个好主意。
如果你试图这样,最好是你的系统提供良好的文件锁定功能,否则服务器将不能正确协作。
如果你将多个服务器同时写入日志文件,你也冒着你的日志文件称为混乱的根源的风险。
在你运行 isamchk 和 myisamchk 时。isamchk 和 myisamchk 实用程序用于表的维护、诊错和
修复,就想你想的那样,因为这些程序可以修改表内容,允许它们与服务器正在操作的同时
对表操作,这样能导致表损坏。理解如何限制这种相互影响是很重要的,这样你不会损坏你
的表。
2.2 数据目表示
每个 mysql 服务器管理的数据库有自己的数据库表,它是数据目录下的一个子目录,其
名字与它表示的数据库相同。例如数据库 my_db 对应于数据库目录 DATADIR/my_db。
这种表示允许多个数据库级的语句在其实现中十分简单。CREATE DATABASE db_name 在
数据目录中创建一个 db_name 空目录,具有只允许 mysql 服务器用户(运行服务器的 Unix 用户)
的属主和模式,这等价于下列手工在服务器主机上创建数据库:
%mkdir DATADIR/db_name%chmod 700 DADADIR/db_name
用一个空目录表示一个新数据库的最简单方法与其它数据库甚至为一个空数据库创建大
量的控制文件或系统文件正好相反。
DROP DATABASE 语句实现同样简单。DROP DATABASE db_name 删除数据库中的 db_name 目
录和所有表文件,这几乎与下列命令一样:
%rm -rf DATADIR/db_name
(差别是服务器只删除具有已知用于表的后缀名的文件。如果你在数据库目录创建了其
它文件。则服务器保留它们,而且目录本身不被删除。
SHOW DATABASE 基本上不做什么,只是列出位于数据目录中的目录名。有些数据库系统
保持一个主表,用于维护所有数据库,但在 mysql 无此构件。由于赋予数据目录结构的简洁性,数据库列表隐含在数据目录的内容中,而且这样的表不必有额外的开销。
2.3 数据库表的表示
每个数据库在数据库目录中有 3 个文件:一个样式(描述文件)、一个数据文件和一个
索引文件。每个文件的基本名是表名,文件名扩展名代表文件类型。扩展名如下表。数据和
索引文件的扩展名指出表使用老式 IASM 索引或新式 MyISAM 索引。表 mysql 文件类型
文件类型 文件名扩展名 文件内容
样式文件 .frm 描述表的结构(它的列、列类型、索引等)。
数据文件 .ISD(ISAM)
或.MYD(MyISAM) 包含数据文件上的所有索引的索引树。
索引文件 .ISM(ISAM)
或.MYI(MyISAM) 该索引文件依赖表是否有索引而存在。
当你发出一条 CREATE TABLE tbl_name 时语句定义表的结构时,服务器创建一个名为
tbl_name.frm 的文件,它包括该结构的内部编码,同时也创建一个空数据和索引文件,初始
化为包含指出无记录和无索引的信息(如果 CREATE TABLE 语句包括索引指定,索引文件反映
出这些索引)。对应于表的文件的属主和模式被设置为只允许 mysql 服务器用户访问。
当你发出一条 ALTER TABLE tbl_name 语句时,服务器重新编码 tbl_name.frm,并修改数
据和索引文件的内容以反映语句指定的结构改变。对于 CREATE INDEX 和 DROP INDEX 也是一样,
因为它们被服务器视为与 ALTER TABLE 等价。DROP TABLE 通过删除对应于表的三个文件来实
现。
虽然你可以通过删除数据库目录中对应于表的三个文件,但不能手工创建或修改一个表,
如,如果 my_db 是当前数据库,DROP TABLE my_tbl 大概等价于下列命令。
% rm -rf DATADIR/my_db/my_tbl.*
SHOW TABLE my_db 的输出只是列出 my_db 数据库目录中的.frm 文件的基文件名。有些数
据库系统一个注册表,列举所有包含在一个数据库中的表,mysql 不是,因为不必要,注册
表 隐含在数据目录的结构中。

2.4 操作系统对数据库和表命名的限制
mysql 对命名数据库和表有一个原则:名字可以由当前字符集中的任何字母数字字符组成,
下划线和美元符 $ 也可以。名字最长为 64 个字符。
然而,因为数据库和表的名字对应于目录和文件名,服务器运行的操作系统可能强加额
外的限制。
首先,数据库和表名仅限于对文件名合法的字符,如 $ 在 mysql 的原则中是允许的,但是
如果你的操作系统不允许,则你不能在目录或表名中使用它。实际上,这对 Unix 或 Windows
不是所担心的,最大的难度是在执行数据库管理时直接在 shell 中引用名字,例如,如果你
命名一个数据库如 $my_db,包含一个美元符,任何从 shell 中对该名字的引用可能被 shell
解释为对一个变量的引用:
%ls $my_db
my_db:undefined variable
对此,你必须转义 $ 字符或用引号禁止其特殊含义:
%ls $my_db
%ls $my_db
如果你用引号,一定要用单引号,而双引号并不禁止变量解释。
其次,虽然 mysql 允许数据库和表名最长到 64 个字符,但名字的长度受限于你的操作系
统限定的长度,一般这不是一个问题(虽然老的 System V 强制 14 个字符)。在这种情况下,
你数据库名的上限为 14 个字符,而表名上限为 10 个字符,因为表示表的文件名有一个点(.)
和三个字符的扩展名。
第三,文件系统的大小写敏感性影响到你如何命名和引用数据库和表名。如果文件系统
是大小写敏感的(如 Unix),两个名字 my_tbl 和 MY_TBL 是不同的表。如果文件系统不是大小
写敏感的(如 Windows),这两个名字指的是相同的表。如果你用一个 Unix 服务器开发数据
库,并且如果你有可能转移到 Windows,你应该记住这一点。

2.5 mysql 状态文件
除了数据库目录,mysql 数据目录还包含很多状态文件,这些文件总结在下表中。大多
数文件的缺省名从服务器主机名生成,在下表中表示为 HOSTNAME。表 mysql 状态文件
文件类型 缺省名 文件内容
进程 ID HOSTNAME.pid 服务器进程的 ID
出错日志 HOSTNAME.err 启动和关闭事件和出错情况
一般日志 HOSTNAME.log 连接 / 断开事件和查询信息
更新日志 HOSTNAME.nnn 修改表结构级内容的所有查询文本

当服务器启动时,它将其进程 ID 写入进程 ID(PID)文件中,而在它关闭时,删除该文件。
PID 文件是允许服务器本身被其他进程找到的工具。例如,如果你运行 mysql.server,在系统
关闭时,关闭 mysql 服务器的脚本检查 PID 文件以决定它需要向哪个进程发出一个终止信号。
出错日志由 safe_mysqld 创建,作为服务器标准出错输出的重定向,它包含任何发到 stderr
的消息。这意味着只有你通过调用 safe_mysqld 启动服务器,出错文件才存在(无论如何,
它是一个启动服务器的最好方法,因为如果它由于出错而退出,safe_mysqld 将重启服务器。)。
一般日志和更新日志是可选的。你可以只开启你需要的日志类型,用 –log 和
–log-update 服务器选项。
一般日志提供服务器操作的一般信息:谁从哪里连接服务器和他们发出什么查询。更新
日志提供查询信息,但只有修改数据库内容的查询。更新日志内容被写成 SQL 语句,可以将
它们提供给 mysql 客户程序来执行。如果你遇上崩溃,并且必须倒回备份文件,更新日志就
很有用,因为你能重复执行自崩溃时的更新,通过将更新日志反馈给服务器,这允许你将数
据库恢复到崩溃发生时的状态。
下面是一个简单的例子,信息出现在一般日志中,它是一个创建一个在数据库 test 中表,
插入一行,然后删除表的会话:
990509 7:37:09 492 Connect Paul@localhost on test 492 Query show databases
492 Query show tables 492 Field List tbl_1 492 Field List tbl_2 …990509
7:34:22 492 Query CREATE TABLE my_tbl (val INT)990509 7:34:34 492 Query
INSERT INTO my_tbl values (1)990509 7:34:38 492 Query DROP TABLE my_tbl
990509 7:34:40 492 Quit

一般日志包含日期和时间、服务器进程 ID、事件类型和事件信息栏目。
同一个会话出现在更新日志中看上去像这样:
use test;CREATE TABLE my_tbl (val int);INSERT INTO my_tbl VALUES(1);
DROP TABLE my_tbl;

对更新日志,用 –log-long-format 选项获得一个扩展形式的日志,扩展日志提供有关
谁何时发出每一条查询,这使用更多的磁盘空间,但如果你想知道谁在做什么,而不用将更
新日志对照一般日志的内容找到连接事件。
对上面的会话,扩展更新日志产生这样的信息:
# Time: 990507 7:32:42# User@Host: paul [paul] @ localhost []use test;
CREATE TABLE my_tbl (val int);# User@Host: paul [paul] @ localhost
[]INSERT INTO my_tbl VALUES(1);# Time: 990507 7:32:43#
User@Host: paul [paul] @ localhost []DROP TABLE my_tbl;

保证你的日志文件安全并且不让任意用户读取是个好主意。一般日志和更新日志都能包
含诸如口令等的敏感信息,因为它们包含查询文本。如:
990509 7:23:31 4 Query UPDATE user SET Password=PASSWORD(secret) WHERE user= root

对于检查和设置数据目录的权限,请见《mysql 安全性指南》。使数据目录安性的指令
包含下列命令:
% chmod 700 DATADIR
以拥有数据目录的 Unix 用户运行此命令。确保服务器也以此用户运行,否则该命令不仅
将其它人拒之门外,它也阻止服务器访问你的数据库。
状态文件出现在数据目录的顶级目录,就象数据库目录,所以你可能担心这些文件名是
否与数据库名冲突或出错(如在服务器执行 SHOW DATABASES 语句时)。答案是不。状态和日
志文件信息存储在文件中,而数据库是目录,所以可执行程序能用一个简单的 stat() 调用区
分它们。如果你看一下数据目录,你可以区分状态文件和数据库目录,用 ls - l 并检查模式
的第一个字符是一个 _ 还是一个 d。
你也可以简单地看一下名字,所有状态文件名包含一个点(.),而数据库目录没有
(. 在数据库名中是无效字符)。

MySQL 数据目录结构(2)

3 重定位数据库目录
前面讨论的数据目录结构是缺省配置,所有数据库和状态文件均包含其中,然而,你有
某些自由决定数据目录内容的位置,本节讨论为什么你可能移走部分数据目录(或甚至目录
本身)、你能移走什么以及你如何做这些改变。
MySQL 允许你重定位数据目录或其中的成员,由几个原因你为什么要这样做:
你能将数据目录放在你缺省所在的文件系统更大容量的文件系统上。
如果你的数据目录在一个繁忙的硬盘上,你可能把它放在不太忙的磁盘上以均衡磁盘活
动。你可以把数据库和日志文件放在分开的磁盘上或跨磁盘分布。
你可能想运行多个服务器,各自有自己的数据目录,这是解决每个进程文件描述符限制
问题的一种方法,特别是你不能重新配置内核以允许更高的限制。
有些系统在例如 /var/run 中保存服务器的部分文件,你可能想把 MySQL 的 PID 文件也放在
那儿,为了系统操作的一致性。

3.1 重定位方法
有两种方法重定位数据目录的内容:
你可以在服务器启动时指定选项,在命令行或在一个选项文件的[mysqld] 中。
你可以移走要重定位的东西,然后在原位置做一个指向新位置的符号连接。

两种方法都不能解决你能重定位的一切,下表总结了什么能重定位和用哪种方法重定位。
如果你使用选项文件,有可能在全局选项文件 /etc/my.cnf(Windows 上的 c:my.cnf)指定
选项。当前的 Windows 版本也寻找系统目录(c:Windows 或 c:NT)。表 重定位方法
重定位方法 适用的重定位方法
整个数据目录 启动选项或符号连接
单个数据库目录 符号连接
单个数据库表 符号连接
PID 文件 启动选项
一般日志 启动选项
更新日志 启动选项

你也可以使用缺省数据目录中的选项文件 my.cnf,但不推荐使用该文件。如果你想重定
位数据目录本身,你不得不让缺省数据目录可读以便使你能在这里放置选项文件指定服务器
应该在哪里找到“真正”的数据目录!这很混乱。如果你想使用一个选项文件指定服务器选项,
最好使用 /etc/my.cnf。

3.1 检验重定位的效果
在试图重定位任何东西之前,检验操作达到预期效果是个好主意。借助于 du、df 和
ls - l 命令获得磁盘空间的信息,但这些依赖于你正确了解你的文件系统的布局。
下面演示一个在你检验一个属目录重定位时的设计陷阱。假定你的数据目录是
/usr/local/var,而你想把它移到 /var/mysql,因为 df 显示 /var 文件系统有很多的空闲空间:
%df /usr /varFilesystem
1k-blocks Used Avail Capacity Mounted on/dev/wd0s3e
396895 292126 73018 80% /usr/dev/wd0s3f
1189359 1111924 162287 15% /var

重定位的数据目录在 /usr 文件系统上有多少空闲空间呢?要知道它,使用 du - s 找出该目录使用多少空间。
%cd /usr/local/var%du -s .133426

这大约是 130MB,真实这样吗?在数据目录下试一下 df:
%df /usr/local/varFilesystem
1k-blocks Used Avail Capacity Mounted on/dev/wd0s3f
1189359 1111924 162287 15% /var

这就奇怪了。如果我们为包含 /usr/local/var 的文件系统申请空闲空间,为什么却报告
var 上的空间呢?这里 ls - l 提供了答案:
%ls -l /usr/local….lrwxrwxrwx 1 root wheel 10 Dec 11 23:33 var – /var/mysql….

输出显示 /usr/local/var 是对 /var/mysql 的符号连接,换句话说,数据目录已经被重定
位于 /var 文件系统,并用一个指向那里的符号连接代替。通过将数据目录移到 /var 竟然释放
了 /usr 上那么多空间!

3.2 重定位数据目录

要重定位数据目录,关闭服务器并把数据目录移到新位置上,然后你应该删除员数据目
录并用指向新位置的符号连接代替它,或用明确指出新位置的选项重启服务器。下表列出指
定位置的命令行和选项。表 数据目录重定位语法
选项源 语法
命令行 –data-dir=/path/to/dir
选项文件 [mysqld]
datadir=/path/to/dir

3.3 重定位数据库

数据库能通过符号连接的方法移走。要重定位一个数据库,关闭服务器并移走数据库目
录并删除原来的数据库目录,用指向新位置的符号连接代替它,然后重启服务器。
下例显示你如何将一个数据库 bigdb 移到一个不同的地方:
%mysqladmin -u root -p shutdownEnter password: ******
%cd DATADIR%tar cf – bigdb | (cd /var/db; tar xf -)
%mv bigdb bigdb.origln -s /var/db/bigdb .
%safe_mysqld
你应该以该数据目录的拥有者执行这些命令。为了安全起见,原数据库目录改名为
bigdb.orig。在你验证了服务器工作正常后,你可以删除原数据目录。
%rm -rf bigdb.orig

3.4 重定位数据库表

重定位一个单独的表不是个好主意。你可以通过把表文件移到一个不同地方,并在数据
目录中创建指向这些文件的符号连接进行。然而,如果你发出一条
ALTER TABLE 或 OPTIMIZE TABLE 语句,将不进行你的修改。
每个语句通过在数据库目录中创建一个实现你修改或优化的临时表,然后删除原来的表
并将临时表更名为原来的表来完成,结果是你的符号连接被删除,而且新表又回到数据库目
录,这是你移走前的原表文件位置。更糟糕的是,你还没有意识到它们在那儿,继续占据着
空间,而且符号连接已经被破坏,这样以后当你意识到发生的事情时,如果你忘记你把它们
移到什么地方,你可能没有好办法追踪文件了。
因为很难保证具有表访问权的人不修改或优化表,所以最好把表留在数据库目录中。

3.5 重定位状态文件

你可以重中定位 PID 文件、一般日志和更新日志。出错日志由 safe_mysqld 用启动选项创
建,而不能被重定位(除非你编辑 safe_mysqld)。
要在一个不同位置写入状态文件,关闭服务器,然后由指定新状态文件位置的适当选项
启动它。下表列出每一个文件的命令行和选项文件的语法。表 状态文件重定位语法
选项源 语法
命令行 –pid-file=pidfile
–log=lodfile
–log-update=updatefile
选项文件 [mysqld]
pid-file=pidfile
log=lodfile
log-update=updatefile

如果你用绝对路径名指定状态文件,用该路径创建文件,否则文件在数据目录下创建。
如,如果你指定 –pid-file=/var/run/mysqld.pid,PID 文件是 /var/run/mysqld.pid。如果
你指定 -pid-file=mysqld.pid,PID 文件是 DATADIR/mysqld.pid。
如果你指定无扩展名的更新日志文件,MySQL 在它每次打开更新日志时产生顺序名。这些
名字用一个扩展名.nnn,这里.nnn 是还没被现有更新日志使用的第一个数字(如 update.000,
update.001 等)。你可以通过明确指定扩展名来覆盖顺序名,这时服务器将只使用指定的名字。

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注丸趣 TV 行业资讯频道,感谢您对丸趣 TV 的支持。

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-07-19发表,共计10377字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)