zoukankan      html  css  js  c++  java
  • MyISAM和InnoDB的区别

    MyISAM和InnoDB的差异

    Mysql支持很多表类型的表(即存储引擎),如MyISAM、InnoDB、Memory、Archive、Example等,主要介绍两种MyISAM和InnoDB。

    MyISAM是默认表类型,基于ISAM(Indexed Sequential Access Method索引顺序访问方法),它是存储记录和文件的标准方法,不是事务安全的,不支持外键,如果有大量的select,MyISAM比较合适。MyISAM表示独立于操作系统之外的,通俗点说就是你可以很轻松的将MyISAM表从windows移植到linux或者从linux移植到windows。

    InnoDB支持事务安全,支持外键、行锁,事务是它的最大优点,如果有大量的insert和update,建议用InnoDB,特使是针对高并发和QPS(query per second)较高的情况。

    1.  表锁差异

    MyISAM

      MyISAM只支持表锁,在select、insert、update、delete都会锁表,开销小,加锁快,不会出现死锁。锁定粒度大,发生锁冲突的概率最高,并发量最低。

    InnoDB

      InnoDB支持事务和行锁,开销大,加锁慢,会出现死锁。锁力度小,发生锁冲突的概率小,并发度最高。

      1>     事务的ACID属性

        a)         原子性(Atomicity):原子性是指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。

        b)         一致性(Consistency):事务前后数据的完整性必须保持一致。

        c)         隔离性(Isolation):事务的隔离性是多个用户并发访问数据库时,数据库为每一个用户开启的事务,不能被其他事务的操作数据所干扰,多个并发事务之间要相互隔离。

        d)         持久性(Durability):持久性是指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来即使数据库发生故障也不应该对其有任何影响。

      2>     并发事务带来的问题

        a)         脏读:一个事务修改了还没提交,另一个事务就读了。

        b)         不可重复读:一个事务两次读取数据之间,另一个事务修改了数据。

        c)         幻读:一个事务两次读取数据之间,另一个事务新增了数据。

        d)         丢失修改(更新丢失):两个事务同时修改每个数据,一个事务的提交破坏了另一个事务的修改。例如,T1读A修改为A-1,T2也读A修改为A-1(实际上T2读A-1修改为A-2)。

      3>     事务隔离级别

    隔离级别

    读数据一致性

    并发副作用

    脏读

    不可重复读

    幻读

    未提交读

    (Read uncommitted)

    最低级别,不读物理上损坏的数据

    已提交读

    (Read committed)

    语句级

    可重复读

    (Repeatable read)

    事务级

    可序列化

    (Serializable)

    最高级别,

    事务级

        查看mysql的默认事务隔离级别“show global variables like ‘tx_isolation’;”

      4>     InnoDB的行锁模式有以下几种:共享锁,排他锁,意向共享锁(表锁),意向排他锁(表锁),间隙锁。

        注意:当语句没有使用索引,InnoDB不能确定操作的行,这个时候就使用的意向锁,也就是表锁。

      5>     关于死锁

        a)         什么是死锁?当两个事务都需要获得对方持有的排他锁才能完成事务,这样就导致了循环锁等待,也就是常见的死锁类型。

        b)         解决死锁的方法:

          1、数据库参数;

          2、 应用中尽量约定程序读取表的顺序一样;

          3、 应用中处理一个表时,尽量对处理的顺序排序;

          4、 调整事务隔离级别(避免两个事务同时操作一行不存在的数据,容易发生死锁)。

    2.  数据库文件差异

    MyISAM

      MyISAM属于堆表。

      1>     在磁盘上存储有3个文件,每个文件以表名为开头。

        .frm用于存储表的定义

        .MYD用于存放表的数据

        .MYI用于存放表的索引

      2>     在mysql中可以创建3种MyISAM格式的表——静态、动态和压缩。格式不需要单独指定,mysql会根据表结构自动选择最合适的格式。

        a)         MyISAM静态:如果表的每个字段的数据类型的定义都是使用静态的(如char),mysql就会自动使用静态MyISAM格式,这种类型格式的表的性能是很高的,也就是查询更新用的时间很少,但要知道这是在牺牲空间为代价。因为每一列都要分配最大的空间,即使有部分空间没有用到,这就使得静态的表所占的空间会比较大。

        b)         MyISAM动态:如果表的每个字段的数据类型的定义都是使用动态的(如varchar),mysql就会自动使用动态MyISAM格式,这种类型格式的表的性能会有所下降,但是它的空间占有要比静态的少很多。

        c)          MyISAM压缩:如果有一张表在设计之初只赋予了它读的使命,就可以用MyISAM压缩表,在相同的配置下,它的性能是最快的。

    InnoDB

      InnoDB属于索引组织表。

      InnoDB有两种存储方式,共享表空间存储和多表空间存储,两种存储方式的表结构和MyISAM一样,以表名开头,扩展名是.frm。

      如果使用共享表空间,那么所有表的数据文件和索引文件都保存在一个表空间里,一个表空间可以有多个文件,通过innodb_data_file_path和innodb_data_home_dir参数设置共享表空间的位置和名字,一般共享表空间的名字叫ibdata1-n。

      如果使用多表空间,那么每个表都有一个表空间文件用于存储每个表的数据和索引,文件名以表名开头,以.ibd为扩展名。

    3.  索引差异

    关于自动增长

    MyISAM引擎的自动增长列必须是索引,如果是组合索引,自动增长可以不是第一列,他可以根据前面几列进行排序后递增。

    InnoDB引擎的自动增长列必须是索引,如果是组合索引也必须是组合索引的第一列。

    关于主键

    MyISAM允许没有任何索引和主键的表存在,MyISAM的索引都是保存行的地址。

    InnoDB引擎如果没有设定主键或者非空唯一索引,就会自动生成一个6字节的主键(用户不可见),InnoDB的数据是主索引的一部分,附加索引保存的是主索引的值。

    关于count()函数

    MyISAM保存有表的总行数,如果select count(*) from table;会直接取出出该值。

    InnoDB没有保存表的总行数,如果使用select count(*) from table;就会遍历整个表,消耗相当大,但是在加了wehre 条件后,MyISAM和InnoDB处理的方式都一样。

    全文索引

    MyISAM支持 FULLTEXT类型的全文索引。

    InnoDB不支持FULLTEXT类型的全文索引,但是InnoDB可以使用sphinx插件支持全文索引,并且效果更好。(sphinx   是一个开源软件,提供多种语言的API接口,可以优化mysql的各种查询)。

    delete from table

    使用这条命令时,InnoDB不会从新建立表,而是一条一条的删除数据,在InnoDB上如果要清空保存有大量数据的表,最好不要使用这个命令。(推荐使用truncate table,不过需要用户有drop此表的权限)。

    索引保存位置

    MyISAM的索引以表名+.MYI文件分别保存。

    InnoDB的索引和数据一起保存在表空间里。

    4.  几点注意事项

    1. 可以用 show create table tablename 命令看表的引擎类型。

    2. 对不支持事务的表做start/commit操作没有任何效果,在执行commit前已经提交。

    3. 可以执行以下命令来切换非事务表到事务(数据不会丢失),InnoDB表比MyISAM表更安全:alter table tablename type=innodb;或者使用 alter table tablename engine = innodb;

    4. 默认innodb是开启自动提交的,如果你按照MyISAM的使用方法来编写代码页不会存在错误,只是性能会很低。如何在编写代码时候提高数据库性能呢?

        1>       尽量将多个语句绑到一个事务中,进行提交,避免多次提交导致的数据库开销。

        2>       在一个事务获得排他锁或者意向排他锁以后,如果后面还有需要处理的sql语句,在这两条或者多条sql语句之间程序应尽量少的进行逻辑运算和处理,减少锁的时间。

        3>       尽量避免死锁。

        4>       sql语句如果有where子句一定要使用索引,尽量避免获取意向排他锁。

        5>       针对开发,一般日志表之类的都是不修改的,可以用MyISAM,而其他表要用到事务,要频繁修改的可以用InnoDB。

  • 相关阅读:
    css常用格式
    css选择器
    D1-mini esp8266的资料备份
    总结esp8266刷Python的完整的步骤(终极总结)
    用Python利用pyFirmata控制Arduino实现Blink
    microPython 的逗比报错的问题
    python+opencv+Face++实现人脸识别比对
    TypeError: slice indices must be integers or None or have an __index__ method
    opencv识别验证码的教程和资料
    校园网一键登录后台
  • 原文地址:https://www.cnblogs.com/xc-chejj/p/11245034.html
Copyright © 2011-2022 走看看