Heim >Datenbank >MySQL-Tutorial >Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

WBOY
WBOYnach vorne
2023-05-30 15:14:441653Durchsuche

Vorwort

Aus Gründen der Datensicherheit muss die Datenbank regelmäßig gesichert werden. Beim Sichern der Datenbank hat man jedoch am meisten Angst, da dies am wahrscheinlichsten zu Dateninkonsistenzen führt gibt ein einfaches Beispiel, das sich jeder ansehen kann:

Angenommen, ein Benutzer gibt während des Datenbanksicherungszeitraums eine Bestellung auf, dann können die folgenden Probleme auftreten:

  • Inventartabelle mit Abzug des Inventars.

  • Inventartabelle sichern.

  • Sichern Sie die Bestelltabellendaten.

  • Fügen Sie Bestellungen zum Bestellformular hinzu.

  • Die Benutzertabelle zieht den Kontostand ab.

  • Benutzertabelle sichern.

Wenn Sie der oben genannten Logik folgen, fehlt in der Bestelltabelle in der Sicherungsdatei ein Datensatz. Wenn Sie diese Sicherungsdatei zum Wiederherstellen von Daten verwenden, fehlt ein Datensatz, was zu Dateninkonsistenzen führt.

Um dieses Problem zu lösen, bietet MySQL viele Lösungen, die wir einzeln erklären und ihre Vor- und Nachteile analysieren.

1. Vollständige Datenbank schreibgeschützt

Um dieses Problem zu lösen, können wir uns vorstellen, dass die Datenbank während der Datenbanksicherung schreibgeschützt ist und nicht Die Methode, die gesamte Datenbank schreibgeschützt zu machen, ist ebenfalls sehr einfach. Zuerst führen wir die folgende SQL aus und schauen uns den Wert der entsprechenden Variablen an:

show variables like 'read_only';

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

Das ist möglich Sehen Sie, dass read_only standardmäßig AUS ist, was der geschlossene Zustand ist. Führen Sie zunächst das folgende SQL aus: read_only 是 OFF,即关闭状态,我们先把它改为 ON,执行如下 SQL:

set global read_only=1;

1 表示 ON,0 表示 OFF,执行结果如下:

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

这个 read_only 对 super 用户无效,所以设置完成后,接下来我们退出来这个会话,然后创建一个不包含 super 权限的用户,用新用户登录,登录成功之后,执行一个插入 SQL,结果如下:

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

可以看到,这个错误信息中说,现在的 MySQL 是只读的(只能查询),不能执行当前 SQL。

加了只读属性,就不用担心备份的时候发生数据不一致的问题了。

但是 read_only 我们通常用来标识一个 MySQL 实例是主库还是从库:

  • read_only=0,表示该实例为主库。数据库管理员 DBA 可能每隔一段时间就会对该实例写入一些业务无关的数据来判断主库是否可写,是否可用,这就是常见的探测主库实例是否活着的。

  • read_only=1,表示该实例为从库。通常,对从库进行定期探活时,只会执行一些读操作,例如执行 "select 1;" 这样的语句。

所以,read_only 这个属性其实并不适合用来做备份,而且如果使用了 read_only 属性将整个库设置为 readonly 之后,如果客户端发生异常,则数据库就会一直保持 readonly 状态,这样会导致整个库长时间处于不可写状态,风险很高。

因此这种方案不合格。

2. 全局锁

全局锁,顾名思义,就是把整个库锁起来,锁起来的库就不能增删改了,只能读了。

那么我们看看怎么使用全局锁。MySQL 提供了一个加全局读锁的方法,命令是 flush tables with read lock (FTWRL)。当你需要让整个库处于只读状态的时候,可以使用这个命令,之后其他线程的增删改等操作就会被阻塞。

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

从图中可以看到,使用 flush tables with read lock; 指令可以锁定表;使用 unlock tables; 指令则可以完成解锁操作(会话断开时也会自动解锁)。

和第一小节的方案相比,FTWRL 有一点进步,即:执行 FTWRL 命令之后如果客户端发生异常断开,那么 MySQL 会自动释放这个全局锁,整个库回到可以正常更新的状态,而不会一直处于只读状态。

但是!!!

加了全局锁,就意味着整个数据库在备份期间都是只读状态,那么在数据库备份期间,业务就只能停摆了。

所以这种方式也不是最佳方案。

3. 事务

不知道小伙伴们是否还记得松哥之前和大家分享的数据库的隔离级别,四种隔离级别中有一个是可重复读(REPEATABLE READ)

mysqldump -h localhost -uroot -p123 test08 > test08.sql
1 bedeutet EIN, 0 bedeutet AUS, das Ausführungsergebnis ist wie folgt:

Wie stellt MySQL die Konsistenz von Sicherungsdaten sicher🎜 🎜Dieser read_only ist für den Superuser nicht gültig, daher verlassen wir die Sitzung und erstellen eine neue Sitzung. Benutzer mit Superberechtigungen melden sich nach erfolgreicher Anmeldung an , führen Sie eine SQL-Einfügung aus und die Ergebnisse sind wie folgt: 🎜🎜Wie stellt MySQL sicher? die Konsistenz der Sicherungsdaten“ />🎜🎜Wie Sie sehen können, besagt diese Fehlermeldung, dass das aktuelle MySQL schreibgeschützt ist (nur Abfragen durchführen kann) und das aktuelle SQL nicht ausführen kann. 🎜🎜Durch das Hinzufügen des schreibgeschützten Attributs müssen Sie sich während der Sicherung keine Sorgen über Dateninkonsistenzen machen. 🎜🎜Aber <code>read_only</code> wird normalerweise verwendet, um zu identifizieren, ob eine MySQL-Instanz die Hauptbibliothek oder eine Slave-Bibliothek ist: 🎜🎜🎜🎜read_only=0, was angibt, dass die Instanz die Hauptbibliothek ist. Der Datenbankadministrator DBA schreibt möglicherweise von Zeit zu Zeit einige nicht geschäftliche Daten in die Instanz, um festzustellen, ob die Hauptbibliothek beschreibbar und verfügbar ist. Dies ist eine übliche Methode, um festzustellen, ob die Hauptbibliotheksinstanz aktiv ist. 🎜🎜🎜🎜read_only=1, was anzeigt, dass die Instanz eine Slave-Bibliothek ist. Normalerweise werden beim regelmäßigen Durchsuchen der Slave-Bibliothek nur einige Lesevorgänge ausgeführt, z. B. Anweisungen wie „select 1;“. 🎜🎜🎜🎜Das Attribut <code>read_only</code> ist also nicht für die Sicherung geeignet, und wenn das Attribut <code>read_only</code> verwendet wird, um die gesamte Bibliothek auf schreibgeschützt zu setzen, tritt beim Client eine Ausnahme auf, Die Datenbank bleibt im schreibgeschützten Zustand, was dazu führt, dass sich die gesamte Bibliothek für lange Zeit in einem nicht beschreibbaren Zustand befindet und das Risiko sehr hoch ist. 🎜🎜Diese Lösung ist also nicht qualifiziert. 🎜🎜2. Globale Sperre🎜🎜Globale Sperre sperrt, wie der Name schon sagt, die gesamte Bibliothek. Die gesperrte Bibliothek kann nicht hinzugefügt, gelöscht oder geändert, sondern nur gelesen werden. 🎜🎜Dann sehen wir uns an, wie man die globale Sperre verwendet. MySQL bietet eine Methode zum Erhöhen der globalen Lesesperre. Der Befehl lautet <code>Tabellen mit Lesesperre leeren</code> (FTWRL). Wenn Sie die gesamte Bibliothek in einen schreibgeschützten Zustand versetzen müssen, können Sie diesen Befehl verwenden. Anschließend werden Vorgänge wie Hinzufügungen, Löschungen und Änderungen durch andere Threads blockiert. 🎜🎜<img src=🎜🎜Von der Bild Wie Sie sehen können, kann die Tabelle mit dem Befehl flush tables with read lock; gesperrt werden. Der Entsperrvorgang kann mit dem Befehl unlock tables; abgeschlossen werden wird auch automatisch entsperrt, wenn die Sitzung getrennt wird. 🎜🎜Im Vergleich zur Lösung im ersten Abschnitt weist FTWRL eine leichte Verbesserung auf, d kann normal aktualisiert werden und bleibt nicht schreibgeschützt. 🎜🎜Aber! ! ! 🎜🎜Das Hinzufügen einer globalen Sperre bedeutet, dass sich die gesamte Datenbank während des Sicherungszeitraums in einem schreibgeschützten Zustand befindet, sodass das Geschäft nur während des Datenbanksicherungszeitraums gestoppt werden kann. 🎜🎜Diese Methode ist also nicht die beste Lösung. 🎜🎜3. Transaktionen🎜🎜Ich weiß nicht, ob ihr euch noch an die Isolationsstufen der Datenbank erinnert, die Brother Song zuvor mit euch geteilt hat. Dies ist REPEATABLE READ auch die Standardisolationsstufe von MySQL. 🎜🎜Wenn ein Benutzer dieselbe SELECT-Anweisung mehrmals in einer anderen Transaktion unter dieser Isolationsstufe ausführt, ist das Ergebnis immer dasselbe. (Da die durch die ausgeführte Transaktion generierten Datenänderungen für die Außenwelt nicht sichtbar sind.) 🎜

换言之,在 InnoDB 这种支持事务的存储引擎中,那么我们就可以在备份数据库之前先开启事务,此时会先创建一致性视图,然后整个事务执行期间都在用这个一致性视图,而且由于 MVCC 的支持,备份期间业务依然可以对数据进行更新操作,并且这些更新操作不会被当前事务看到。

在可重复读的隔离级别下,即使其他事务更新了表数据,也不会影响备份数据库的事务读取结果,这就是事务四大特性中的隔离性,这样备份期间备份的数据一直是在开启事务时的数据。

具体操作也很简单,使用 mysqldump 备份数据库的时候,加上 -–single-transaction 参数即可。

为了看到 -–single-transaction 参数的作用,我们可以先开启 general_loggeneral_log 即 General Query Log,它记录了 MySQL 服务器的操作。当客户端连接、断开连接、接收到客户端的 SQL 语句时,会向 general_log 中写入日志,开启 general_log 会损失一定的性能,但是在开发、测试环境下开启日志,可以帮忙我们加快排查出现的问题。

通过如下查询我们可以看到,默认情况下 general_log 并没有开启:

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

我们可以通过修改配置文件 my.cnf(Linux)/my.ini(Windows),在 mysqld 下面增加或修改(如已存在配置项)general_log 的值为1,修改后重启 MySQL 服务即可生效。

也可以通过在 MySQL 终端执行 set global general_log = ON 来开启 general log,此方法可以不用重启 MySQL

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

开启之后,默认日志的目录是 mysql 的 data 目录,文件名默认为 主机名.log

接下来,我们先来执行一个不带 -–single-transaction 参数的备份,如下:

mysqldump -h localhost -uroot -p123 test08 > test08.sql

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

大家注意默认的 general_log 的位置。

接下来我们再来加上 -–single-transaction 参数看看:

mysqldump -h localhost -uroot -p123 --single-transaction test08 > test08.sql

Wie stellt MySQL die Konsistenz der Sicherungsdaten sicher?

大家看我蓝色选中的部分,可以看到,确实先开启了事务,然后才开始备份的,对比不加 -–single-transaction 参数的日志,多了开启事务这一部分。

Das obige ist der detaillierte Inhalt vonWie stellt MySQL die Konsistenz der Sicherungsdaten sicher?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:yisu.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen