深入理解MySQL中的事務(wù)機(jī)制
使用數(shù)據(jù)庫(kù)事務(wù)可以確保除事務(wù)性單元內(nèi)的所有操作都成功完成。MySQL中的InnoDB引擎的表才支持transaction。在一個(gè)事務(wù)里,如果出現(xiàn)一個(gè)數(shù)據(jù)庫(kù)操作失敗了,事務(wù)內(nèi)的所有操作將被回滾,數(shù)據(jù)庫(kù)將會(huì)回到事務(wù)前的初始狀態(tài)。有一些不能被回滾的語句:將在本文的最后討論。
在一個(gè)web應(yīng)用中,會(huì)很經(jīng)常遇到需要使用事務(wù)的地方,要么希望若干語句都執(zhí)行成功,要么都不執(zhí)行,如果出現(xiàn)有些執(zhí)行成功,而其他的失敗將會(huì)導(dǎo)致數(shù)據(jù)損壞。
在這篇文章的例子中,我們使用下面的兩張表"employee"和"telephone",下面是SQL語句(作為參考):
創(chuàng)建 employee表:
CREATE TABLE `employee` ( `id` int NOT NULL AUTO_INCREMENT, `first_name` varchar(100) NOT NULL, `last_name` varchar(100) NOT NULL, `job_title` varchar(100) DEFAULT NULL, `salary` double DEFAULT NULL, `notes` text, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
向employee中插入數(shù)據(jù)
INSERT INTO `employee` (`first_name`, `last_name`, `job_title`, `salary`) VALUES ('Robin', 'Jackman', 'Software Engineer', 5500), ('Taylor', 'Edward', 'Software Architect', 7200), ('Vivian', 'Dickens', 'Database Administrator', 6000), ('Harry', 'Clifford', 'Database Administrator', 6800), ('Eliza', 'Clifford', 'Software Engineer', 4750), ('Nancy', 'Newman', 'Software Engineer', 5100), ('Melinda', 'Clifford', 'Project Manager', 8500), ('Harley', 'Gilbert', 'Software Architect', 8000);
創(chuàng)建telephone表
CREATE TABLE `telephone` ( `id` int NOT NULL AUTO_INCREMENT, `employee_id` int DEFAULT NULL, `type` varchar(20) NOT NULL, `no` varchar(50) NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
向telephone表插入數(shù)據(jù)
INSERT INTO `telephone` (`employee_id`, `type`, `no`) VALUES (1, 'mobile', '245-249697'), (2, 'mobile', '270-235969'), (2, 'land', '325-888885'), (3, 'mobile', '270-684972'), (4, 'mobile', '245-782365'), (4, 'land', '325-888886'), (5, 'mobile', '245-537891'), (6, 'mobile', '270-359457'), (7, 'mobile', '245-436589'), (7, 'land', '325-888887'), (8, 'mobile', '245-279164'), (8, 'land', '325-888888');
設(shè)想你需要一個(gè)新的叫做Grace Williams雇員,并帶有他的電話號(hào)碼信息。你可能會(huì)執(zhí)行下面兩句sql:
INSERT INTO `employee` (`id`, `first_name`, `last_name`, `job_title`, `salary`) VALUES (9, 'Grace', 'Williams', 'Softwaree Engineer', 5000); INSERT INTO `telephone` (`id`, `employee_id`, `type`, `no`) VALUES (13, 9, 'mobile', '270-598712');
讓我們看看第二個(gè)語句,在第一個(gè)語句中,employee_id是在第一條語句中指定的,設(shè)想一下,當(dāng)?shù)谝粭l語句失敗,而第二條語句成功的狀況。在這種狀況下,telephone表中就會(huì)有一條employee_id為9的記錄,而employee表中并沒有id為9的記錄,而如果將這兩個(gè)語句放在MySQL事務(wù)中,如果第一條語句失敗,那么第二條語句也將回滾,從而不會(huì)造成這種問題。
在PHP(PHP參考文檔)中我們可以使用如下的方式啟用事務(wù):
<?php //$salary = 5000; $salary = '$5000'; /* Change database details according to your database */ $dbConnection = mysqli_connect('localhost', 'robin', 'robin123', 'company_db'); mysqli_autocommit($dbConnection, false); $flag = true; $query1 = "INSERT INTO `employee` (`id`, `first_name`, `last_name`, `job_title`, `salary`) VALUES (9, 'Grace', 'Williams', 'Softwaree Engineer', $salary)"; $query2 = "INSERT INTO `telephone` (`id`, `employee_id`, `type`, `no`) VALUES (13, 9, 'mobile', '270-598712')"; $result = mysqli_query($dbConnection, $query1); if (!$result) { $flag = false; echo "Error details: " . mysqli_error($dbConnection) . ". "; } $result = mysqli_query($dbConnection, $query2); if (!$result) { $flag = false; echo "Error details: " . mysqli_error($dbConnection) . ". "; } if ($flag) { mysqli_commit($dbConnection); echo "All queries were executed successfully"; } else { mysqli_rollback($dbConnection); echo "All queries were rolled back"; } mysqli_close($dbConnection); ?>
當(dāng)你執(zhí)行mysqli_query函數(shù)的時(shí)候,結(jié)果被立即提交到了數(shù)據(jù)庫(kù)。使用mysqli_autocommit函數(shù),可以關(guān)閉自動(dòng)提交,執(zhí)行結(jié)果只有當(dāng)你想提交的時(shí)候才提交。
如果任何語句執(zhí)行失敗我們都可以設(shè)置$flag變量為false。如果有很多語句要執(zhí)行,可以考慮將他們放在for循環(huán)中。
最后,如果flag是true(也就是沒有錯(cuò)誤發(fā)生),我們使用mysqli_commit提交事務(wù)。否則我們使用mysqli_rollback回滾事務(wù)。
所以,事務(wù)可以在某種程度上幫助我們維護(hù)數(shù)據(jù)的完整和正確,另外,為了保證數(shù)據(jù)無誤,我們還推薦使用外鍵。
并不是所有的語句都是支持事務(wù)的,例如,如果使用CREATE TABLE或者ALTER TABLE語句,需要了解更多可以參考MySQL手冊(cè)查看哪些語句不能回滾。
相關(guān)文章
為什么MySQL數(shù)據(jù)庫(kù)索引選擇使用B+樹?
今天小編就為大家分享一篇關(guān)于為什么MySQL數(shù)據(jù)庫(kù)索引選擇使用B+樹?,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-03-03CentOS 6.2 安裝 MySQL 5.7.28的教程(mysql 筆記)
本文通過圖文并茂的形式給大家介紹了CentOS 6.2 安裝 MySQL 5.7.28的教程,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2019-11-11MySQL自動(dòng)為查詢數(shù)據(jù)結(jié)果加序號(hào)
這篇文章主要給大家介紹了關(guān)于MYSQL如何自動(dòng)為查詢數(shù)據(jù)的結(jié)果編上序號(hào)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用mysql具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起看看吧2022-12-12計(jì)算機(jī)管理服務(wù)中找不到mysql的服務(wù)的解決辦法
MySQL是一種流行的開源關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng),用于存儲(chǔ)和管理大量數(shù)據(jù),在計(jì)算機(jī)管理中,啟動(dòng)MySQL服務(wù)是一項(xiàng)重要的任務(wù),因?yàn)樗梢源_保數(shù)據(jù)庫(kù)系統(tǒng)的順利運(yùn)行,這篇文章主要給大家介紹了關(guān)于計(jì)算機(jī)管理服務(wù)中找不到mysql的服務(wù)的解決辦法,需要的朋友可以參考下2023-05-05老鳥帶你開發(fā)專業(yè)規(guī)范的MySQL啟動(dòng)腳本
這篇文章主要介紹了老鳥帶你開發(fā)專業(yè)規(guī)范的MySQL啟動(dòng)腳本,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09mysql備份恢復(fù)mysqldump.exe幾個(gè)常用用例
收集了,一個(gè)整理不錯(cuò)的,mysql備份與恢復(fù)用法2008-08-08