加入收藏 | 设为首页 | 会员中心 | 我要投稿 聊城站长网 (https://www.0635zz.com/)- 智能语音交互、行业智能、AI应用、云计算、5G!
当前位置: 首页 > 站长学院 > MySql教程 > 正文

部分关于MySQL事务的基础知识

发布时间:2023-11-10 16:45:29 所属栏目:MySql教程 来源:
导读:下面讲讲关于MySQL事务的基础知识,文字的奥妙在于贴近主题相关。所以,闲话就不谈了,我们直接看下文吧,相信看完MySQL事务的基础知识这篇文章你一定会有所受益。

1、事务的基本语法

<!--创建表-->

mys
下面讲讲关于MySQL事务的基础知识,文字的奥妙在于贴近主题相关。所以,闲话就不谈了,我们直接看下文吧,相信看完MySQL事务的基础知识这篇文章你一定会有所受益。
 
1、事务的基本语法
 
<!--创建表-->
 
mysql> create table bank
 
    -> (
 
    -> name varchar(25),
 
    -> money float
 
    -> );
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> insert into bank values('lu','1000'),('qi','5000');   <!--插入数据-->
 
Query OK, 2 rows affected (0.00 sec)
 
Records: 2  Duplicates: 0  Warnings: 0
 
mysql> begin;        <!--begin开启事务,start transaction也可开启事务-->
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> update bank set money=money - 1000 where name='qi';
 
Query OK, 1 row affected (0.00 sec)
 
Rows matched: 1  Changed: 1  Warnings: 0
 
mysql> update bank set money=money+1000 where name ='lu';
 
Query OK, 1 row affected (0.00 sec)
 
Rows matched: 1  Changed: 1  Warnings: 0
 
mysql> select * from bank;           <!--查看数据-->
 
+------+-------+
 
| name | money |
 
+------+-------+
 
| lu   |  2000 |
 
| qi   |  4000 |
 
+------+-------+
 
2 rows in set (0.00 sec)
 
mysql> rollback;              <!--回滚事务-->
 
Query OK, 0 rows affected (0.01 sec)
 
mysql> select * from bank;           <!--再次查询数据,发现已经便会了原来的值-->
 
+------+-------+
 
| name | money |
 
+------+-------+
 
| lu   |  1000 |
 
| qi   |  5000 |
 
+------+-------+
 
2 rows in set (0.00 sec)
 
mysql> commit;            <!--提交事务-->
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from bank;          <!--查询数据-->
 
+------+-------+
 
| name | money |
 
+------+-------+
 
| lu   |  1000 |
 
| qi   |  5000 |
 
+------+-------+
 
2 rows in set (0.00 sec)
 
一个事务所涉及到的命令如下:
 
事务开始:start transaction或begin;
 
事务提交:commit
 
回滚:rollback
 
查看自动提交模式是自动还是手动
 
mysql> show variables like 'AUTOCOMMIT';
 
+---------------+-------+
 
| Variable_name | Value |
 
+---------------+-------+
 
| autocommit    | ON    |          <!--“ON”表示自动提交-->
 
+---------------+-------+
 
1 row in set (0.01 sec)
 
mysql> set AUTOCOMMIT=0;         <!--关闭自动提交,0是关闭,1是开启-->
 
mysql> show variables like 'AUTOCOMMIT';            <!--再次查看-->
 
+---------------+-------+
 
| Variable_name | Value |
 
+---------------+-------+
 
| autocommit    | OFF   |
 
+---------------+-------+
 
1 row in set (0.00 sec)
 
2、事务的四种隔离级别
 
一些关于MySQL事务的基础知识
 
事务在提交之前对其他事务可不可见。
 
read unaommitted(未提交读)
 
read committed(已提交读)
 
Repeatable read(可重复读)
 
seaializable(可串行化)
 
1)未提交读
 
事务中修改没有提交对其他事务也是可见的,俗称脏读。
 
<!--创建一个测试表-->
 
mysql> create table student
 
    -> (
 
    -> id int not null auto_increment,
 
    -> name varchar(32) not null default '',
 
    -> primary key(id)
 
    -> )engine=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;
 
接下来需要自行开启两个MySQL会话终端,A和B,并且都执行以下命令设置为未提交读。
 
mysql> set session tx_isolation='read-uncommitted';
 
客户端A:
 
mysql> begin;
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from student;
 
Empty set (0.00 sec)
 
mysql> insert into student(name) values('zhangyi');
 
<!--注意,此时事务未提交!!!-->
 
mysql> set session tx_isolation='read-uncommitted';        <!--设置为未提交读-->
 
Query OK, 0 rows affected, 1 warning (0.00 sec)
 
mysql> select * from student;     <!--查询表,即可看到客户A没有提交的事务-->
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
+----+---------+
 
1 row in set (0.00 sec)
 
总结:以上可以看出未提交读隔离级别非常危险,对于一个没有提交事务所做修改对另一个事务是可见状态,出现了脏读!非特殊情况不建议使用此级别。
 
2)已提交读
 
多数数据库系统默认为此级别(MySQL不是)。已提交读级别为一个事务只能已提交事务所做的修改,也就是解决了未提交读的问题。
 
客户端A插入数据测试:
 
mysql> set session tx_isolation='read-committed';
 
Query OK, 0 rows affected, 1 warning (0.00 sec)
 
mysql> begin;
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
+----+---------+
 
1 row in set (0.00 sec)
 
mysql> insert into student(name) values('zhanger');
 
Query OK, 1 row affected (0.00 sec)
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhanger |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
客户端B查看(不会看到客户端A插入的数据):
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
+----+---------+
 
1 row in set (0.00 sec)
 
客户端A进行提交:
 
mysql> commit;
 
Query OK, 0 rows affected (0.01 sec)
 
客户端B进行查看(就可以看到A插入的数据了):
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhanger |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
总结:从上面可以看出,提交读没有了未提交读的问题,但是我们可以看到客户端A的一个事务中执行了两次同样的SELECT语句,得到不同的结果,因此已提交读又被称为不可重复读。同样的筛选条件可能得到不同的结果。
 
3)可重复读
 
可重复读解决了不可重复读的问题,数据库级别没有解决幻读的问题。
 
以下是客户端A和客户端B同时操作(都设置为可重复读,然后两边都开启一个事务):
 
mysql> set session tx_isolation='repeatable-read';
 
Query OK, 0 rows affected, 1 warning (0.00 sec)
 
mysql> begin;
 
Query OK, 0 rows affected (0.00 sec)
 
客户端A:
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhangsi |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
mysql> update student set name='zhanger' where id=3;
 
Query OK, 1 row affected (0.01 sec)
 
Rows matched: 1  Changed: 1  Warnings: 0
 
mysql> commit;
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from student;           
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhanger |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
客户端B:
 
mysql> select * from student;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhangsi |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
mysql> commit;         <!--提交当前事务-->
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from student;         <!--即可看到客户端A更新的数据-->
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhanger |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
总结:上面可以看出,可重复读两次读取的内容不一样。数据库的幻读问题并没有得到解决。幻读只读锁定里面的数据,不能读锁定外的数据,解决幻读出了mvcc机制Mvcc机制。
 
4)可串行化
 
是最高隔离级别,强制事务串行执行,执行串行了也就解决问题了,这个只有在对数据一致性要求非常严格并且没有并发的情况下使用。
 
在客户端A及客户端B进行以下操作(设置为可串行读):
 
mysql> set session tx_isolation='serializable';
 
客户端A:
 
mysql> begin;
 
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from student where id < 10;
 
+----+---------+
 
| id | name    |
 
+----+---------+
 
|  2 | zhangyi |
 
|  3 | zhanger |
 
+----+---------+
 
2 rows in set (0.00 sec)
 
客户端B:
 
mysql> insert into student(name) values('zhangqi');
 
<!--此时进行插入操作时,会一直卡在这里,然后出现下面的报错信息,除非客户端Acommit提交事务-->
 
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
 
 

(编辑:聊城站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章