- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有一个由 MYISAM 表组成的数据库模式,我有兴趣不时从某些表中删除旧记录。
我知道 delete 不会回收内存空间,但是正如我在 DELETE 命令的描述中发现的,插入可能会重用删除的空间
在 MyISAM 表中,已删除的行在链表中维护,后续的 INSERT 操作会重用旧行位置。
如果 LOAD DATA 命令也重用已删除的空间,我很感兴趣?
更新
我也很感兴趣索引空间是如何回收的?
更新 2012-12-03 23:11
根据@RolandoMySQLDBA 的回答提供了更多信息
执行以下建议的查询后,对于需要重用或回收空间的不同表,我得到了不同的结果:
SELECT row_format FROM information_schema.tables
WHERE table_schema='mydb' AND table_name='mytable1';
> 动态
SELECT row_format FROM information_schema.tables
WHERE table_schema='mydb' AND table_name='mytable2';
> 固定
更新 2012-12-09 08:06
LOAD DATA
当且仅当行格式是固定的或(行格式是动态的并且有删除的行大小完全相同)。
看来如果row_format是动态的,对每条记录都会在删除列表中进行全量查找,如果没有找到准确的行大小,则不使用删除的记录,表内存占用会增加, 此外 LOAD DATA
将花费更多时间来导入记录。
我将排除此处给出的答案,因为它完美地描述了所有过程。
最佳答案
对于名为 mydb.mytable
的 MySQL 表,只需运行以下命令:
OPTIMIZE TABLE mydb.mytable;
您也可以分阶段进行:
CREATE TABLE mydb.mytable_new LIKE mydb.mytable;
ALTER TABLE mydb.mytable_new DISABLE KEYS;
INSERT INTO mydb.mytable_new SELECT * FROM mydb.mytable;
ALTER TABLE mydb.mytable_new ENABLE KEYS;
ALTER TABLE mydb.mytable RENAME mydb.mytable_old;
ALTER TABLE mydb.mytable_new RENAME mydb.mytable;
ALTER TABLE mydb.mytable_old;
ANALYZE TABLE mydb.mytable;
无论哪种情况,表最终都没有碎片。
试一试!!!
如果您担心通过 LOAD DATA INFILE
批量插入时行是否被重用,请注意以下几点:
当您创建 MyISAM 表时,我假设默认的行格式是动态的。你可以检查它是什么
SHOW CREATE TABLE mydb.mytable\G
或
SELECT row_format FROM information_schema.tables
WHERE table_schema='mydb' AND table_name='mytable';
由于您的表的行格式是动态
,因此碎片行的大小各不相同。 MyISAM 存储引擎会一直检查每个已删除的行长度,以查看下一组插入的数据是否适合。如果传入数据无法容纳任何已删除的行,则会附加新行数据。
The presence of such rows can make myisamchk
struggle .
这就是我建议运行 OPTIMIZE TABLE
的原因。这样,可以更快地附加数据。
您还可以做一些有趣的事情:Try setting concurrent_insert to 2 .这样,您总是在不检查表中的间隙的情况下追加到 MyISAM 表。这将显着加快 INSERT 的速度,但不会影响所有已知的差距。
您仍然可以尽早使用 OPTIMIZE TABLE
对您的表进行碎片整理。
为什么不运行我的第二个建议
CREATE TABLE mydb.mytable_new LIKE mydb.mytable;
ALTER TABLE mydb.mytable_new DISABLE KEYS;
INSERT INTO mydb.mytable_new SELECT * FROM mydb.mytable;
ALTER TABLE mydb.mytable_new ENABLE KEYS;
ALTER TABLE mydb.mytable RENAME mydb.mytable_old;
ALTER TABLE mydb.mytable_new RENAME mydb.mytable;
ANALYZE TABLE mydb.mytable;
这会给你一个想法
OPTIMIZE TABLE
运行需要多长时间.MYD
和 .MYI
在运行 OPTIMIZE TABLE
后会小多少运行我的第二个建议后,您可以将它们与
SELECT
A.mydsize,B.mydsize,A.mydsize - B.mydsize myd_diff,
A.midsize,B.myisize,A.myisize - B.myisize myi_diff
FROM
(
SELECT data_length mydsize,index_length myisize
FROM information_schema.tables
WHERE table_schema='mydb' AND table_name='mytable'
) A,
(
SELECT data_length mydsize,index_length myisize
FROM information_schema.tables
WHERE table_schema='mydb' AND table_name='mytable_new'
) B;
任何 ROW_FORMAT 设置为固定的表都可以每次分配相同长度的行。如果 MyISAM 表维护一个已删除行的列表,则列表中的第一行应始终被选为下一行以插入数据。在找到具有足够长度的合适行间隙之前,无需遍历整个列表。每个删除的行都会在 DELETE
之后快速追加。每个 INSERT 都会选择已删除行的第一行。
我们可以假设这些事情,因为 MyISAM tables can do concurrent inserts .为了通过 concurrent_insert 提供此功能选项,插入 MyISAM 表必须能够检测到三 (3) 个事物之一:
为了使检测 #1 尽可能快,MyISAM 表的 row_format 必须固定。如果是Dynamic,很有可能需要遍历列表。
关于mysql - LOAD DATA 删除后回收磁盘空间,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13685949/
我在服务器启动时创建一个缓存(服务器启动每次都需要10分钟)。目前我正在使用内存缓存(Ehcache)。现在我想建立一个机制,以便一旦数据被缓存我应该能够在几秒钟内启动服务器。比如将缓存的持久副本写入
我编写 json 结构的方式使得文件(在进行了一个月的测量后)存储在磁盘上时仍然只有 100 MB 左右。但是现在文件大约是 20mb,但我看到我的脚本需要的内存大约是 200/300 mb。显然,脚
Solaris9 x86下如何挂载和永久挂载windows fat32分区 临时挂载Shell 命令; mout –F pcfs /dev/dsk/c1d0p0:c /mnt/c mount
磁盘ID中的资源组名称大小写不敏感。重现此问题的步骤 - 在 Azure 中创建独立磁盘,检查 ID。对于例如 -“/subscriptions/subscriptionID/resourceGrou
我已将附加数据磁盘的备份还原到新虚拟机。当我发出命令 sudo blkid 时,我发现它与附加到原始虚拟机的数据磁盘具有相同的 UUID,因此我无需更改 fstab 即可在启动时挂载它。然而,它似乎是
在用户态中,执行磁盘 IO 就像链接 C 库一样简单,或者,如果您喜欢冒险,可以直接执行系统调用。我想知道内核本身是如何执行 IO 的。 换句话说,假设我在裸机上以特权模式运行应用程序。我将如何访问通
我已将附加数据磁盘的备份还原到新虚拟机。当我发出命令 sudo blkid 时,我发现它与附加到原始虚拟机的数据磁盘具有相同的 UUID,因此我无需更改 fstab 即可在启动时挂载它。然而,它似乎是
我正在尝试使用 laravel 和 ffmpeg 创建缩略图。但是我收到了这个错误。 磁盘 [视频] 没有配置驱动程序。 我的代码 public function index() { FFMp
我的目标是读/写 usb。 首先必须打开并读取 usb 低级别,如“程序” 我使用 visual c++ 和 winAPI 下面是我的测试代码 char path[64]; sprintf(path,
内核缓冲区缓存何时为空?这似乎不是 LINE Buffering。如果我写 () 一个没有换行符的字符串,它会立即输出到文件。 另外,socket文件的输入输出缓冲区是否也像Disk I/O一样使用内
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a software
我有一个大型调用中心,有 250 个并发调用。队列日志的队列应用程序平面文件。该系统使用 Asterisk 和 Queuemetrics。两个服务都在同一台服务器上运行。规范为 16 核和 64 GB
我在使用安装了 Centos7 的 VMWare VM 时遇到问题。 lsblk 命令给出如下内容 df -h 给出这个 我正在尝试将 root lvm 扩展到分区,但无论我如何尝试都无法做到这一点。
在基于内存的计算模型中,通过考虑数据结构,可以抽象地完成唯一需要进行的运行时计算。 但是,关于高性能磁盘 I/O 算法的文档并不多。因此,我提出了以下一组问题: 1) 我们如何估计磁盘 I/O 操作的
我不是在寻找调用命令行实用程序的代码,它可以解决问题。我实际上很想知道用于创建 RAM 磁盘的 API。 编辑 动机:我有一个第三方库,它需要一个目录名,以便以某种方式处理该目录中的文件。我将这些文件
MySQL 数据库显示磁盘 I/O 利用率持续保持在 100% 左右。数据库服务器有 24 GB 内存。 我们尝试优化查询,但效果不佳。 请检查如下所示的当前配置参数: 参数 当前值 key_buff
这是交易。我们本可以采用完全静态 html 的方式来解决性能问题,但由于该站点将是部分动态的,因此这对我们来说行不通。我们想到的是使用 memcache + eAccelerator 来加速 PHP
对于游戏 Minecraft,运行服务器应用程序时的一般方法是在 RAMDisk 中运行它,因为它使用数百个小文件来生成世界,I/O 速度是主要瓶颈。 在最近的尝试中,我尝试使用 Dokan/ImDi
当我查找文件中的某个位置并写入少量数据(20 字节)时,幕后发生了什么? 我的理解 据我所知,可以从磁盘写入或读取的最小数据单位是一个扇区(传统上是 512 字节,但该标准现在正在改变)。这意味着要写
如何使用golang获取xen服务器的内存、磁盘、网络和cpu信息? 是否有任何可用的软件包? 最佳答案 与其他服务器有什么不同?如果没有 - 有一堆 Go 包可以做到这一点,我正在使用这个 - ht
我是一名优秀的程序员,十分优秀!