下面講講關(guān)于MySQL事務(wù)的基礎(chǔ)知識,文字的奧妙在于貼近主題相關(guān)。所以,閑話就不談了,我們直接看下文吧,相信看完MySQL事務(wù)的基礎(chǔ)知識這篇文章你一定會有所受益。
溫宿網(wǎng)站建設(shè)公司創(chuàng)新互聯(lián),溫宿網(wǎng)站設(shè)計制作,有大型網(wǎng)站制作公司豐富經(jīng)驗。已為溫宿上千余家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\成都外貿(mào)網(wǎng)站建設(shè)要多少錢,請找那個售后服務(wù)好的溫宿做網(wǎng)站的公司定做!
<!--創(chuàng)建表-->
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'); <!--插入數(shù)據(jù)-->
Query OK, 2 rows affected (0.00 sec)
Records: 2 Duplicates: 0 Warnings: 0
mysql> begin; <!--begin開啟事務(wù),start transaction也可開啟事務(wù)-->
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; <!--查看數(shù)據(jù)-->
+------+-------+
| name | money |
+------+-------+
| lu | 2000 |
| qi | 4000 |
+------+-------+
2 rows in set (0.00 sec)
mysql> rollback; <!--回滾事務(wù)-->
Query OK, 0 rows affected (0.01 sec)
mysql> select * from bank; <!--再次查詢數(shù)據(jù),發(fā)現(xiàn)已經(jīng)便會了原來的值-->
+------+-------+
| name | money |
+------+-------+
| lu | 1000 |
| qi | 5000 |
+------+-------+
2 rows in set (0.00 sec)
mysql> commit; <!--提交事務(wù)-->
Query OK, 0 rows affected (0.00 sec)
mysql> select * from bank; <!--查詢數(shù)據(jù)-->
+------+-------+
| name | money |
+------+-------+
| lu | 1000 |
| qi | 5000 |
+------+-------+
2 rows in set (0.00 sec)一個事務(wù)所涉及到的命令如下:
- 事務(wù)開始:start transaction或begin;
- 事務(wù)提交:commit
- 回滾:rollback
mysql> show variables like 'AUTOCOMMIT'; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | autocommit | ON | <!--“ON”表示自動提交--> +---------------+-------+ 1 row in set (0.01 sec) mysql> set AUTOCOMMIT=0; <!--關(guān)閉自動提交,0是關(guān)閉,1是開啟--> mysql> show variables like 'AUTOCOMMIT'; <!--再次查看--> +---------------+-------+ | Variable_name | Value | +---------------+-------+ | autocommit | OFF | +---------------+-------+ 1 row in set (0.00 sec)

事務(wù)在提交之前對其他事務(wù)可不可見。
- read unaommitted(未提交讀)
- read committed(已提交讀)
- Repeatable read(可重復(fù)讀)
- seaializable(可串行化)
事務(wù)中修改沒有提交對其他事務(wù)也是可見的,俗稱臟讀。
<!--創(chuàng)建一個測試表--> 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,并且都執(zhí)行以下命令設(shè)置為未提交讀。
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');
<!--注意,此時事務(wù)未提交!!!-->mysql> set session tx_isolation='read-uncommitted'; <!--設(shè)置為未提交讀--> Query OK, 0 rows affected, 1 warning (0.00 sec) mysql> select * from student; <!--查詢表,即可看到客戶A沒有提交的事務(wù)--> +----+---------+ | id | name | +----+---------+ | 2 | zhangyi | +----+---------+ 1 row in set (0.00 sec)
總結(jié):以上可以看出未提交讀隔離級別非常危險,對于一個沒有提交事務(wù)所做修改對另一個事務(wù)是可見狀態(tài),出現(xiàn)了臟讀!非特殊情況不建議使用此級別。
多數(shù)數(shù)據(jù)庫系統(tǒng)默認為此級別(MySQL不是)。已提交讀級別為一個事務(wù)只能已提交事務(wù)所做的修改,也就是解決了未提交讀的問題。
客戶端A插入數(shù)據(jù)測試:
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插入的數(shù)據(jù)):
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插入的數(shù)據(jù)了):
mysql> select * from student; +----+---------+ | id | name | +----+---------+ | 2 | zhangyi | | 3 | zhanger | +----+---------+ 2 rows in set (0.00 sec)
總結(jié):從上面可以看出,提交讀沒有了未提交讀的問題,但是我們可以看到客戶端A的一個事務(wù)中執(zhí)行了兩次同樣的SELECT語句,得到不同的結(jié)果,因此已提交讀又被稱為不可重復(fù)讀。同樣的篩選條件可能得到不同的結(jié)果。
可重復(fù)讀解決了不可重復(fù)讀的問題,數(shù)據(jù)庫級別沒有解決幻讀的問題。
以下是客戶端A和客戶端B同時操作(都設(shè)置為可重復(fù)讀,然后兩邊都開啟一個事務(wù)):
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; <!--提交當(dāng)前事務(wù)--> Query OK, 0 rows affected (0.00 sec) mysql> select * from student; <!--即可看到客戶端A更新的數(shù)據(jù)--> +----+---------+ | id | name | +----+---------+ | 2 | zhangyi | | 3 | zhanger | +----+---------+ 2 rows in set (0.00 sec)
總結(jié):上面可以看出,可重復(fù)讀兩次讀取的內(nèi)容不一樣。數(shù)據(jù)庫的幻讀問題并沒有得到解決。幻讀只讀鎖定里面的數(shù)據(jù),不能讀鎖定外的數(shù)據(jù),解決幻讀出了mvcc機制Mvcc機制。
是最高隔離級別,強制事務(wù)串行執(zhí)行,執(zhí)行串行了也就解決問題了,這個只有在對數(shù)據(jù)一致性要求非常嚴格并且沒有并發(fā)的情況下使用。
在客戶端A及客戶端B進行以下操作(設(shè)置為可串行讀):
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');
<!--此時進行插入操作時,會一直卡在這里,然后出現(xiàn)下面的報錯信息,除非客戶端Acommit提交事務(wù)-->
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction| 隔離級別 | 臟讀 | 不可重復(fù) | 幻讀 | 加鎖讀 |
|---|---|---|---|---|
| 未提交讀 | 是 | 是 | 是 | 否 |
| 提交讀 | 否 | 是 | 是 | 否 |
| 可重復(fù)讀 | 否 | 否 | 是 | 否 |
| 串行讀 | 否 | 否 | 否 | 是 |
對于以上MySQL事務(wù)的基礎(chǔ)知識相關(guān)內(nèi)容,大家還有什么不明白的地方嗎?或者想要了解更多相關(guān),可以繼續(xù)關(guān)注我們的行業(yè)資訊板塊。
新聞名稱:一些關(guān)于MySQL事務(wù)的基礎(chǔ)知識
轉(zhuǎn)載來于:http://chinadenli.net/article38/pgpdpp.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供軟件開發(fā)、企業(yè)建站、網(wǎng)站改版、定制開發(fā)、企業(yè)網(wǎng)站制作、網(wǎng)站設(shè)計公司
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)