位置: 编程技术 - 正文

MySQL下使用Inplace和Online方式创建索引的教程(mysql in())

编辑:rootadmin

推荐整理分享MySQL下使用Inplace和Online方式创建索引的教程(mysql in()),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:mysql in的使用,mysql语句in的用法,mysql使用replace,mysql inodb,mysql inodb,mysql语句in的用法,mysql语句in的用法,mysql in的使用,内容如对您有帮助,希望把文章链接给更多的朋友!

MySQL各版本,对于add Index的处理方式是不同的,主要有三种:

(1)Copy Table方式这是InnoDB最早支持的创建索引的方式。顾名思义,创建索引是通过临时表拷贝的方式实现的。

新建一个带有新索引的临时表,将原表数据全部拷贝到临时表,然后Rename,完成创建索引的操作。

这个方式创建索引,创建过程中,原表是可读的。但是会消耗一倍的存储空间。

(2)Inplace方式这是原生MySQL 5.5,以及innodb_plugin中提供的创建索引的方式。所谓Inplace,也就是索引创建在原表上直接进行,不会拷贝临时表。相对于Copy Table方式,这是一个进步。

Inplace方式创建索引,创建过程中,原表同样可读的,但是不可写。

(3)Online方式这是MySQL 5.6.7中提供的创建索引的方式。无论是Copy Table方式,还是Inplace方式,创建索引的过程中,原表只能允许读取,不可写。对应用有较大的限制,因此MySQL最新版本中,InnoDB支持了所谓的Online方式创建索引。

InnoDB的Online Add Index,首先是Inplace方式创建索引,无需使用临时表。在遍历聚簇索引,收集记录并插入到新索引的过程中,原表记录可修改。而修改的记录保存在Row Log中。当聚簇索引遍历完毕,并全部插入到新索引之后,重放Row Log中的记录修改,使得新索引与聚簇索引记录达到一致状态。

与Copy Table方式相比,Online Add Index采用的是Inplace方式,无需Copy Table,减少了空间开销;与此同时,Online Add Index只有在重放Row Log最后一个Block时锁表,减少了锁表的时间。

与Inplace方式相比,Online Add Index吸收了Inplace方式的优势,却减少了锁表的时间。

1.Inplace add Index

测试表

Inplace Add Index处理流程SQL

处理流程

Inplace Add Index实现分析在索引创建完成之后,MySQL Server立即可以使用新建的索引,做查询。但是,根据以上流程,对我个人来说,有三个疑问点:

索引数据字典上,为何需要维护一个trx_id?trx_id有何作用?

遍历聚簇索引读取所有记录时,为何可跳过删除项?只读取非删除项,那么新建索引上没有版本信息,无法处理原有事务的快照读;

MySQL Server层,为何需要等待打开表的只读事务提交?等待当前表上的只读事务,可以保证这些事务不会使用到新建索引

根据分析,等待打开表的只读事务结束较好理解。因为新索引上没有版本信息,若这些事务使用新的索引,将会读不到正确的版本记录。

那么InnoDB是如何处理其他那些在创建索引之前已经开始,但却一直未提交的老事务呢?这些事务,由于前期为并未读取当前表,因此不会被等待结束。这些事务在RR隔离级别下,会读取不到正确的版本记录,因为使用的索引上并没有版本信息。

当然,InnoDB同样考虑到了此问题,并采用了一种比较简介的处理方案。在索引上维护一个trx_id,标识创建此索引的事务ID。若有一个比这个事务更老的事务,打算使用新建的索引进行快照读,那么直接报错。

考虑如下的并发处理流程(事务隔离级别为RR):

当session 1执行最后一条select时,MySQL Optimizer会选择idx_t1_b索引进行查询,但是索引上并没有b = 1的项,使用此索引会导致查询出错。那么,InnoDB是如何处理这个情况的呢?

处理流程:

MySQL Server收到InnoDB返回的错误之后,会将错误报给用户,用户会收到以下错误:

2.Online add Index

测试表

Online Add Index处理流程SQL

处理流程

Online Add Index实现分析在看完前面分析的InnoDB 5.6.7-RC版本中实现的基本处理流程之后,个人仍旧遗留了几个问题,主要的问题有:

MySQL下使用Inplace和Online方式创建索引的教程(mysql in())

Online Add Index是否支持Unique索引?

确切的答案是:支持(不过存在Bug,后面分析)。InnoDB支持Online创建Unique索引。

既然支持,就会面临Check Duplicate Key的问题。Row Log中如果存在与索引中相同的键值怎么处理?怎么检测是否存在相同键值?

InnoDB解决此问题的方案也比较简介易懂。其维护了一个row_merge_dup_t的数据结构,存储了在Row log重放过程中遇到的违反唯一性冲突的Row Log。应用完Row Log之后,外部判断是否存在Unique冲突(有多少Unique冲突,均会记录),Online创建Unique索引失败。

Row Log是什么样的结构,如何组织的?

在Online Add Index过程中,并发DML产生的修改,被记录在Row Log中。首先,Row Log不是InnoDB的Redo Log,而是每个正在被Online创建的索引的独占结构。

Online创建索引,遵循的是先创建索引数据字典,后填充数据的方式。因此,当索引数据字典创建成功之后,新的DML操作就可以读取此索引,尝试进行更新。但是,由于索引结构上的status状态为ONLINE_INDEX_CREATION,因此这些更新不能直接应用到新索引上,而是放入Row Log之中,等待被重放到索引之上。

Row Log中,以Block的方式管理DML操作内容的存放。一个Block的大小为由参数innodb_sort_buffer_size控制,默认大小为1M ()。初始化阶段,Row Log申请两个这样的Block。

在Row Log重放的过程中,到底需要多久的锁表时间?

前面的流程分析中,也提到了锁表的问题(内部为锁新建索引树的操作实现)。

在重放Row log时,有两个情况下,需要锁表:

情况一:在使用完一个Block,跳转到下一个Block时,需要短暂锁表,判断下一个Block是否为Row Log的最后一个Block。若不是最后一个,跳转完毕后,释放锁;使用Block内的row log不加锁,用户DML操作仍旧可以进行。

情况二:在使用最后一个Block时,会一直持有锁。此时不允许新的DML操作。保证最后一个Block重放完成之后,新索引与聚簇索引记录达到一致状态。

综上分析两个锁表情况,情况二会持续锁表,但是由于也只是最后一个Block,因此锁表时间也较短,只会短暂的影响用户操作,在低峰期,这个影响是可以接受的。

3. Online Add Index是否也存在与Inplace方式一样的限制?

由于Online Add Index同时也是Inplace方式的,因此Online方式也存在着Inplace方式所存在的问题:新索引上缺乏版本信息,因此无法为老事务提供快照读。

不仅如此,相对于Inplace方式,Online方式的约束更甚一筹,不仅所有小于创建此Index的事务不可使用新索引,同时,所有在新索引创建过程中开始的事务,也不能使用新索引。

这个增强的限制,在rowmerge.cc::row_merge_read_clustered_index()函数中调整,在聚簇索引遍历完成之后,将新索引的trx_id,赋值为Online Row Log中最大的事务ID。待索引创建完成之后,所有小于此事务ID的事务,均不可使用新索引。

在遍历聚簇索引读取数据时,读取的是记录的最新版本,那么此记录是否在Row Log也会存在?InnoDB如何处理这种情况?

首先,答案是肯定的。遍历聚簇索引读取记录最新版本时,这些记录有可能是新事务修改/插入的。这些记录在遍历阶段,已经被应用到新索引上,于此同时,这些记录的操作,也被记录到Row Log之中,出现了一条记录在新索引上存在,在Row Log中也存在的情况。

当然,InnoDB已经考虑到了这个问题。在重放Row Log的过程中,对于Row Log中的每条记录,首先会判断其在新索引中是否已经存在(row0log.c::row_log_apply_op_low()),若存在,则当前Row Log可以跳过(或者是将操作类型转换)。

例如:Row Log中记录的是一个INSERT操作,若此INSERT记录在新索引中已经存在,那么Row Log中的记录,可以直接丢弃(若存在项与INSERT项完全一致);或者是将INSERT转换为UPDATE操作(Row Log记录与新索引中的记录,部分索引列有不同);

Online Add Index是否存在Bug?

答案同样是肯定的,存在Bug。

其中有一个Bug,重现方案如下:

在以上的测试中,首先为表准备足够的数据,目的是session 1做Online Add Index的读取聚簇索引阶段,session 2新的记录也能够被读到。

在session 1的Online Add Index完成之后(成功),执行以下两个命令,结果如下:

可以看到,b上已经有了一个Unique索引,但是表中却存在两个相同的取值为的值。

此Bug,是处理Row Log的重放过程,未详尽考虑所有情况导致的。因此,在MySQL 5.6版本稳定之前,慎用!

Online Add Index可借鉴之处在MySQL 5.6.7中学习到两个文件操作函数:一是posix_fadvise()函数,指定POSIX_FADV_DONTNEED参数,可做到读写不Cache:Improving Linux performance by preserving Buffer Cache State unbuffered I/O in Linux;二是fallocate()函数,指定FALLOC_FL_PUNCH_HOLE参数,可做到读时清空:Linux Programmer's Manual FALLOCATE(2) 有类似需求的朋友,可试用。

posix_fadvise函数+POSIX_FADV_DONTNEED参数,主要功能就是丢弃文件在Cache中的clean blocks。因此,若用户不希望一个文件占用过多的文件系统Cache,可以定期的调用fdatasync(),然后接着posix_fadvise(POSIX_FADV_DONTNEED),清空文件在Cache中的clean blocks,不错的功能!

MySQL中InnoDB存储引擎的锁的基本使用教程 MyISAM和MEMORY采用表级锁(table-levellocking)BDB采用页面锁(page-levelocking)或表级锁,默认为页面锁InnoDB支持行级锁(row-levellocking)和表级锁,默认为行级

MySQL中二进制与重做日志文件的基本概念学习教程 二进制日志二进制日志记录了所有对数据库执行更改的操作,二进制主要有以下两种作用:1、恢复(recovery)2、复制(replication)二进制日志的启动:

讲解Linux系统下如何自动备份MySQL数据的基本教程 1.先创建一个数据库备份目录:mkdirbackupcdbackupmkdirmysqlbackupcdmysqlbackup2.创建备份脚本vimysqlautobackup3.编写脚本:filename=`date+%Y%m%d`/mysql的bin目录/mysqldump--opt

标签: mysql in()

本文链接地址:https://www.jiuchutong.com/biancheng/312372.html 转载请保留说明!

上一篇:MySQL中使用or、in与union all在查询命令下的效率对比(MySQL中使用_____语句更新表中的数据)

下一篇:MySQL中InnoDB存储引擎的锁的基本使用教程(mysql innodb存储结构)

  • 计提税额与实缴税额的区别是什么?
  • 不是公司的车能抵扣进项吗
  • 应收账款贷方余额怎么调平
  • 代扣代缴增值税纳税义务发生时间
  • 对公账户余额和实际不符
  • 哪个会计科目必须对全部项目发函证
  • 来料加工生产成本账务处理
  • 申请开红字发票,原蓝字发票要退回来吗
  • 签订代销合同的人有没有退货权
  • 付给其他公司的利息怎么做账
  • 事业编制公考
  • 样品不收钱怎么做会计分录
  • 一般纳税人的进项税额计入成本吗
  • 免税发票有几种类型
  • 管理费用属于什么现金流量项目
  • 个体户经营所得怎么申报
  • 未开发土地可否转给子公司
  • 发票经手人和经办人的区别
  • 个人公积金公司这178,个人交多少
  • 合伙人退伙资产清算需要交税吗
  • 业务招待住宿费发票可以抵扣进项税吗
  • Win11怎么隐藏底部
  • 小规模纳税人如何缴纳企业所得税
  • php红包源码
  • 分公司 股东
  • 销售货物支付运杂费要交增值税吗
  • 一只猫在花园里英语
  • 为庆祝中秋而摆放的东西
  • 房抵债权
  • php验证身份证号
  • thinkphp+vue
  • vue 自适应网站ui框架
  • 对抗生成网络算法
  • api接口使用方法
  • springcloud阿里巴巴
  • 增值税纳税申报操作流程
  • 金蝶软件资产负债表怎么生成
  • 国内旅客运输服务
  • 代开专票的个税减免吗?
  • 个体户开出的发票没跟对方说自己冲红了怎么办
  • 抵扣的税可以退掉么?
  • 房地产公司收到客户违约金会计科目
  • 车辆置换差价账务处理
  • 提的税与实际交的税区别
  • 收到银行回单如何打印
  • 合作社注销麻烦吗
  • 个体户查账征收个人所得税税率
  • 淘宝卖家运费险为什么越来越贵
  • 销售收入指开票金额吗
  • 固定资产的税法处理与会计处理的差异50字
  • 计提的意义
  • 极速开票软件怎么开票
  • mysql开发教程
  • Mysql 5.7.19 winx64 ZIP Archive 安装及使用过程问题小结
  • windows性能监视器怎么用
  • 微软官方的网址是多少
  • linux怎么用命令
  • vmware_vmx.exe进程在哪里关
  • win7系统打开浏览器后突然卡住了
  • win8开机进入开始界面
  • xp无法启动如何修复
  • win7专业版系统镜像
  • win8安全和维护在哪
  • linux系统设置网络
  • Cocos2dx3.2 Crazy Tetris 基本设置及主菜单页面(菜单、按钮)
  • python 密码
  • shell 输出
  • linux系统中rpm命令主要功能
  • json web key
  • 批处理怎么学
  • python parser解析
  • js window.screen
  • js实现继承的几种方式详述(推荐)
  • 定额发票查询app
  • 一般纳税人按季申报的行业
  • 江苏发票真伪查询网站
  • 国家减免的增值税怎么做账
  • 国税地税征管体制改革方案
  • 种子在什么情况下能发芽
  • 新疆房车自驾营地
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设