大家都知道在MySQL中,MyISAM比InnoDB查询快,但很多人都不知道其中的原理。

今天我们就来聊聊其中的原理,另外也验证下是否MyISAM比InnoDB真的查询快。

在探索其中原理之前,我们先验证下查询速度。

验证

下面我们一起来验证两个问题:

1、通过主键查询,MyISAM是否比InnoDB查询快?

2、通过非主键字段查询,MyISAM是否比InnoDB查询快?

验证前准备

订单表结构:

CREATE TABLE `trade` (
`id` INT(11) NULL DEFAULT NULL,
`trade_no` INT(11) NULL DEFAULT NULL,
UNIQUE INDEX `id` (`id`),
INDEX `trade_no` (`trade_no`)
)
COMMENT='订单'
COLLATE='utf8_unicode_ci'
ENGINE=InnoDB
;

分别创建不同引擎的表trade_myisam、trade_innodb,并插入10万条测试数据,如下:

这里我们字段都采用Int类型,是为了方便我们,使用区间查询,来验证查询速度。

验证思路

1、分别通过id、trade_no查询10000~20000区间的记录,并获得查询所花费时间。

2、循环查询100次,并取得查询耗费时间平均值。

3、通过对比查询
耗费
时间平均值,来判断性能。

下面我们一起来验证下,文章开头的两个问题:

1、主键查询

C#实现主键查询Id在10000~20000区间,查询耗费时间平均值代码如下:

using (var conn = new MySqlConnection(connsql))
{
conn.Open();
//查询次数
var queryTime = 1000;

//InnoDB数据引擎:
主键查询
//开始时间
var startTime = DateTime.Now;
for (var i = 0; i < queryTime; i++)
{
//插入
var sql = "select * from trade_innodb where id>=10000 and id<20000";
var sqlComm = new MySqlCommand();
sqlComm.Connection = conn;
sqlComm.CommandText = sql;
sqlComm.ExecuteScalar();
sqlComm.Dispose();
}
//完成时间
var endTime = DateTime.Now;
//耗时
var spanTime = (endTime - startTime) * 1.0 / queryTime;
Console.WriteLine("InnoDB主键查询1000次平均耗时:
" + spanTime.Minutes + "分" + spanTime.Seconds + "秒" + spanTime.Milliseconds + "毫秒");

//MyISAM数据引擎:
主键查询
//开始时间
startTime = DateTime.Now;
for (var i = 0; i < queryTime; i++)
{
//插入
var sql = "select * from trade_myisam where id>=10000 and id<20000";
var sqlComm = new MySqlCommand();
sqlComm.Connection = conn;
sqlComm.CommandText = sql;
sqlComm.ExecuteScalar();
sqlComm.Dispose();
}
//完成时间
endTime = DateTime.Now;
//耗时
spanTime = (endTime - startTime) * 1.0 / queryTime;
Console.WriteLine("MyISAM主键查询1000次平均耗时:
" + spanTime.Minutes + "分" + spanTime.Seconds + "秒" + spanTime.Milliseconds + "毫秒");

conn.Close();
}

运行结果:

通过上面的图片,可以看到结果: MyISAM查询性能确实比较Innodb快。

2、非主键查询

C#实现费主键查询trade_no在10000~20000区间,查询耗费时间平均值代码如下:

using (var conn = new MySqlConnection(connsql))
{
conn.Open();
//查询次数
var queryTime = 1000;

//InnoDB数据引擎:
非主键查询
//开始时间
var startTime = DateTime.Now;
for (var i = 0; i < queryTime; i++)
{
//插入
var sql = "select * from trade_innodb where trade_no>=10000 and trade_no<20000";
var sqlComm = new MySqlCommand();
sqlComm.Connection = conn;
sqlComm.CommandText = sql;
sqlComm.ExecuteScalar();
sqlComm.Dispose();
}
//完成时间
var endTime = DateTime.Now;
//耗时
var spanTime = (endTime - startTime) * 1.0 / queryTime;
Console.WriteLine("InnoDB非主键查询1000次平均耗时:
" + spanTime.Minutes + "分" + spanTime.Seconds + "秒" + spanTime.Milliseconds + "毫秒");

//MyISAM数据引擎:
非主键查询
//开始时间
startTime = DateTime.Now;
for (var i = 0; i < queryTime; i++)
{
//插入
var sql = "select * from trade_myisam where trade_no>=10000 and trade_no<20000";
var sqlComm = new MySqlCommand();
sqlComm.Connection = conn;
sqlComm.CommandText = sql;
sqlComm.ExecuteScalar();
sqlComm.Dispose();
}
//完成时间
endTime = DateTime.Now;
//耗时
spanTime = (endTime - startTime) * 1.0 / queryTime;
Console.WriteLine("MyISAM非主键查询1000次平均耗时:
" + spanTime.Minutes + "分" + spanTime.Seconds + "秒" + spanTime.Milliseconds + "毫秒");

conn.Close();
}

运行结果:

通过上面的图片,可以看到结果: MyISAM查询性能确实比较Innodb快。

对比结果

通过我们的验证来看,不管是主键还是非主键查询,MyISAM查询性能确实比较Innodb快。
查询速度不一样,肯定是MyISAM与InnoDB有什么区别,下面我们一起来看看它们的区别。

MyISAM与InnoDB区别

通过上面表格对比, InnoDB在做SELECT的时候,要维护的东西比MYISAM引擎多很多,影响查询速度有:

1)数据块,InnoDB要缓存,MyISAM只缓存索引块,  这中间还有换进换出的减少;

2)InnoDB寻址要映射到块,再到行,MyISAM记录的直接是文件的OFFSET,定位比InnoDB要快

3)InnoDB还需要维护MVCC一致;
虽然你的场景没有,但他还是需要去检查和维护

MVCC (Multi-Version Concurrency Control)多版本并发控制 。

索引区别:

MyISAM与InnoDB索引存储方式是不一样的(关于索引请查看《细说MySql索引原理》)。

InnoDB 的表是根据主键进行展开的 B+tree 的聚集索引。
MyISAM则非聚集型索引,MyISAM存储会有两个文件,一个是索引文件,另外一个是数据文件,其中索引文件中的索引指向数据文件中的表数据。

聚集型索引并不是一种单独的索引类型,而是一种存储方式,InnoDB 聚集型索引实际上是在同一结构中保存了 B+tree 索引和数据行。
当有聚簇索引时,它的索引实际放在叶子页中。

下面我们通过2张图,我们就能直观看到这两个引擎,索引结构的区别:

维护MVCC

InnoDB引擎还有需要实现MVCC,这里面也是损耗一定的性能的。

Innodb是通过每一行记录有2个额外的隐藏值来实现,这两个值一个记录这行数据何时被创建,另外一个记录这行数据何时过期(或者被删除)。

但是InnoDB并不存储这些事件发生时的实际时间,相反它只存储这些事件发生时的系统版本号。
这是一个随着事务的创建而不断增长的数字。
每个事务在事务开始时会记录它自己的系统版本号。
每个查询必须去检查每行数据的版本号与事务的版本号是否相同。

最后

通过我们的验证来看,不管是主键还是非主键查询,MyISAM查询性能确实比较Innodb快。
但是MyISAM不支持事务,在实现业务方面有很多麻烦的事情。

关于我们开发中,选择哪个数据引擎的问题。

平常在我们业务系统中,InnoDB数据引擎完全可以满足我们的需求,这两个引擎查询性能差别还是比较小的,完全没必要为了这点性能,给我们带来更多的工作量。
MyISAM引擎更多是用于存储数据仓库,只有查询少量插入的业务场景中,比如系统日志。

欢迎关注公众号【编程乐趣】,了解更多MySQL知识

MySQL中MyISAM为什么比InnoDB查询快的更多相关文章

  1. MySQL中MyISAM引擎与InnoDB引擎性能简单测试

    [硬件配置]CPU : AMD2500+ (1.8G)内存: 1G/现代硬盘: 80G/IDE[软件配置]OS : Windows XP SP2SE : PHP5.2.1DB : MySQL5.0.3 ...

  2. MySQL中MyISAM引擎及InnoDB引擎的缓存优化设计

    MyISAM引擎中,为了提高io效率以及读取效率,将对磁盘频繁读取的索引数据加载至内存中操作. MyISAM设计了一个在存放在内存中的索引缓冲池Key Cache.Key Cache只缓存索引数据,通 ...

  3. Mysql中MyISAM引擎和InnoDB引擎的比较

    结论 如果不清楚自己应该用什么引擎,那么请选择InnoDB,Mysql5.5+的版本默认引擎都是InnoDB,早期的Mysql版本默认的引擎是MyISAM MyISAM 和 InnoDB的适用场景 M ...

  4. MySQL中MyISAM和InnoDB两种主流存储引擎的特点

    一.数据库引擎(Engines)的概念 MySQ5.6L的架构图: MySQL的存储引擎全称为(Pluggable Storage Engines)插件式存储引擎.MySQL的所有逻辑概念,包括SQL ...

  5. MySQL中MyISAM和InnoDB的区别

    MyISAM和InnoDB的区别 MySQL默认采用的是MyISAM. MyISAM不支持事务,而InnoDB支持.InnoDB的AUTOCOMMIT默认是打开的,即每条SQL语句会默认被封装成一个事 ...

  6. 【mysql中myisam和innodb的区别】

    单击进入源网页 要点摘要: 1.查看mysql存储引擎的状态mysql> show engines; 2.查看mysql默认的存储引擎mysql> show variables like ...

  7. MYSQL 中 MyISAM与InnoDB两者之间区别与选择,详细总结,性能对比

    1.MyISAM:默认表类型,它是基于传统的ISAM类型,ISAM是Indexed Sequential Access Method (有索引的顺序访问方法) 的缩写,它是存储记录和文件的标准方法.不 ...

  8. MySQL中MyISAM与InnoDB的主要区别对比

    特征 MyISAM InnoDB 聚集索引 否 是 压缩数据 是(仅当使用压缩行格式时才支持压缩MyISAM表.使用压缩行格式和MyISAM的表是只读的.) 是 数据缓存 否 是 加密数据 是(通过加 ...

  9. mysql 中 myisam innodb 的区别

    区别: 1. InnoDB支持事务,MyISAM不支持,对于InnoDB每一条SQL语言都默认封装成事务,自动提交,这样会影响速度,所以最好把多条SQL语言放在begin和commit之间,组成一个事 ...

随机推荐

  1. PTA 求链式表的表长

    6-1 求链式表的表长 (10 分)   本题要求实现一个函数,求链式表的表长. 函数接口定义: int Length( List L ); 其中List结构定义如下: typedef struct ...

  2. JetBrains Projector 体验

    先来一张最终效果图: JetBrains Projector 是 JetBrains 的"远程开发"解决方案,基于 Client + Server 架构,对标的是微软 VSCode ...

  3. DAOS 分布式异步对象存储|事务模型

    DAOS API 支持分布式事务,允许将针对属于同一 Container 的对象的任何更新操作组合到单个 ACID 事务中.分布式一致性是通过基于多版本时间戳排序的无锁乐观并发控制机制提供的.DAOS ...

  4. 17. vue-route详细介绍

    前后端路由的来历 前端如何实现页面跳转但是不刷新? 了解hash和history两种方法 vue-router基本使用 安装vue-router 搭建vue-router框架的步骤 vue-route ...

  5. java面试-CAS底层原理

    一.CAS是什么? 比较并交换,它是一条CPU并发原语. CAS是一种无锁算法,CAS有3个操作数,内存值V,旧的预期值A,要修改的新值B.当且仅当预期值A和内存值V相同时,将内存值V修改为B,否则什 ...

  6. 【接入指南】一个Demo带你玩转华为帐号服务

    在<接入指南:一文带你了解华为帐号服务>中已经给大家介绍了华为帐号服务有哪些优势,如一键授权登录华为全场景共享.共享华为帐号所有用户资源.帐号安全可靠.接入方便快捷等,以及为什么能帮助开发 ...

  7. Java中获取类的运行时结构

    获取运行时类的完整结构 通过反射获取运行时类的完整结构 Field(属性).Method(方法).Constructor(构造器).Superclass(父类).Interface(接口).Annot ...

  8. 用递归求n皇后问题

    此问题是指在n*n的国际象棋棋盘上 ,放置n个皇后,使得这n个皇后均不在,同一行,同一列,同一对角线上,求出合法的方案的数目. 本题可以简单转化为就是求n的全排列中的数放在棋盘上使得这几组数,符合均不 ...

  9. inline&register

    inline关键字: 内联只是一个请求,不代表编译器会响应:同时某些编译器会将一些函数优化成为内联函数. C++在类内定义的函数默认是内联函数,具体是否真变成内联函数还需看编译器本身. registe ...

  10. 如果你的application.properties中还存在明文密码----加密Spring Boot中的application.properties

    1 概述 什么?都2020年了还在Spring Boot的配置文件中写明文密码? 虽然是小项目,明文也没人看. 明文简单快捷方便啊!!! 你看直接用户名root密码123456多么简单!!! ... ...