MySQL
数据库的事务
事务: 多条sql语句,要么全部成功,要么全部失败。
事务的特性: 数据库事务特性:原子性(Atomic)、一致性(Consistency)、隔离性(Isolation)、持久性 (Durabiliy)。简称ACID。
- 原子性:组成一个事务的多个数据库操作是一个不可分割的原子单元,只有所有操作都成功, 整个事务才会提交。任何一个操作失败,已经执行的任何操作都必须撤销,让数据库返回初始状态。
- 一致性:事务操作成功后,数据库所处的状态和它的业务规则是一致的。即数据不会被破坏。 如A转账100元给B,不管操作是否成功,A和B的账户总额是不变的。
- 隔离性:在并发数据操作时,不同的事务拥有各自的数据空间,它们的操作不会对彼此产生干扰
- 持久性:一旦事务提交成功,事务中的所有操作都必须持久化到数据库中。
索引是什么
官方介绍索引是帮助MySQL高效获取数据的数据结构。更通俗的说,数据库索引好比是一本书前面的目录,能加快数据库的查询速度。
- 一般来说索引本身也很大,不可能全部存储在内存中,因此索引往往是存储在磁盘上的文件中的(可能存储在单独的索引文件中,也可能和数据一起存储在数据文件中)。
- 我们通常所说的索引,包括聚集索引、覆盖索引、组合索引、前缀索引、唯一索引等,没有特别说明,默认都是使用B+树结构组织(多路搜索树,并不一定是二叉的)的索引。
使用 MySQL 的索引应该注意些什么?
下述操作会导致引擎放弃使用索引而进行全表扫描:
- 在WHERE子句中使用!=或<>操作符,优化器将无法通过索引来确定将要命中的行数,因此需要搜索该表的所有行。
- 在WHERE子句中使用OR来连接条件,如:
SELECT id FROM t WHERE num = 10 OR num = 20
。 - 在WHERE子句中对字段进行表达式操作或者函数操作
- MySQL 评估使用索引比全表扫描更慢
避免出现索引失效:
- 不要在WHERE子句中的=左边进行函数、算术运算或其他表达式运算,如:
SELECT id FROM t WHERE num/2 = 5
- 复合索引遵循最左前缀原则
- 列类型是字符串类型,在查询时要给值加引号
- LIKE查询,%不能在前,因为无法使用索引。如果需要模糊匹配,可以使用全文索引
- 表字段为NULL也是不可以使用索引的
- 字段是字符串类型的使用的时候,必须加引号
SQL优化手段有哪些
- 查询语句中不要使用
select *
, 避免全表扫描,应具体指定字段 - 尽量减少子查询,使用关联查询(left join,right join,inner join)替代
- 减少使用IN或者NOT IN ,使用exists,not exists或者关联查询语句替代
- or 的查询尽量用 union或者union all 代替(在确认没有重复数据或者不用剔除重复数据时, union all会更好)
- 应尽量避免在 where 子句中使用!=或<>操作符,否则引擎会放弃使用索引而进行全表扫描。
- 应尽量避免在 where 子句中对字段进行 null 值判断,否则引擎会放弃使用索引而进行全表 扫描,如:
select id from t where num is null
可以在num上设置默认值0,确保表中num列没有 null 值,然后这样查询:select id from t where num=0
事务
事务隔离级别有哪些?MySQL的默认隔离级别是?
SQL 标准定义了四个隔离级别:
- READ-UNCOMMITTED(读取未提交): 最低的隔离级别,允许读取尚未提交的数据变更,可能会导致脏读、幻读或不可重复读。
- READ-COMMITTED(读取已提交): 允许读取并发事务已经提交的数据,可以阻止脏读,但是幻读或不可重复读仍有可能发生。
- REPEATABLE-READ(可重复读): 对同一字段的多次读取结果都是一致的,除非数据是被本身 事务自己所修改,可以阻止脏读和不可重复读,但幻读仍有可能发生。
- SERIALIZABLE(可串行化): 最高的隔离级别,完全服从ACID的隔离级别。所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读 以及幻读。
MySQL InnoDB 存储引擎的默认支持的隔离级别是 REPEATABLE-READ(可重读)。我们可以通 过SELECT @@tx_isolation;
命令来查看
需要注意的是:与 SQL 标准不同的地方在于 InnoDB 存储引擎在 REPEATABLE-READ(可重读) 事务隔离级别下使用的是Next-Key Lock 锁算法,因此可以避免幻读的产生,这与其他数据库系统(如 SQL Server) 是不同的。所以说InnoDB 存储引擎的默认支持的隔离级别是 REPEATABLEREAD(可重读) 已经可以完全保证事务的隔离性要求,即达到了 SQL标准的 SERIALIZABLE(可串行化) 隔离级别。因为隔离级别越低,事务请求的锁越少,所以大部分数据库系统的隔离级别都是 READ-COMMITTED(读取提交内容) ,但是你要知道的是InnoDB 存储引擎默认使用 REPEAaTABLE-READ(可重读) 并不会有任何性能损失。
InnoDB 存储引擎在分布式事务的情况下一般会用到 SERIALIZABLE(可串行化) 隔离级别。
MySQL事务的ACID是怎么保证的,或者说是怎么实现的
MySQL事务的ACID,其中(C)一致性是最终目的。 保证一致性的措施有:
- A原子性:靠undo log来保证,它记录了需要回滚的日志信息,事务回滚时撤销已经执行成功是sql。
- D持久性:靠redo log来保证,MySQL修改数据的时候会在redo log中记录一份日志数据,就算数据没有保存成功,只要日志保存成功了,数据仍然不会丢失。
- I隔离性:靠数据库的锁,加上MVCC实现的。
- C一致性:一致性是由其他三大特征保证,同时程序代码要保证业务上的一致性。既需要数据库层面保证,又需要应用层面进行保证,并且MySQL底层通过两阶段提交事务保证了事务持久化时的一致性。