HTML Diff
0 added 0 removed
Original 2026-01-01
Modified 2026-03-10
1 <p>Теги: субд, rollback, python, import mysqldb, select, mysql, show processlist, waiting for table metadata lock, autocommit, commit, rocesslist, conn.commit(), conn.autocommit(true), default настройки</p>
1 <p>Теги: субд, rollback, python, import mysqldb, select, mysql, show processlist, waiting for table metadata lock, autocommit, commit, rocesslist, conn.commit(), conn.autocommit(true), default настройки</p>
2 <p>Все мы рано или поздно сталкиваемся с необходимостью сохранить какую-то информацию, да так, чтобы намертво, чтобы внуки потом ещё прочитать смогли. А если ещё нужно делать хитрые выборки по сохранённому, то обычно мы приходим к использованию реляционных СУБД. Чаще всего, если посмотреть рейтинги популярности, это MySQL.</p>
2 <p>Все мы рано или поздно сталкиваемся с необходимостью сохранить какую-то информацию, да так, чтобы намертво, чтобы внуки потом ещё прочитать смогли. А если ещё нужно делать хитрые выборки по сохранённому, то обычно мы приходим к использованию реляционных СУБД. Чаще всего, если посмотреть рейтинги популярности, это MySQL.</p>
3 <h2>Что может быть проще?</h2>
3 <h2>Что может быть проще?</h2>
4 <p>Качаем последний MySQL и запускаем. И вот уже где-то в коде устанавливается соединение с БД, выполняется простой запрос:</p>
4 <p>Качаем последний MySQL и запускаем. И вот уже где-то в коде устанавливается соединение с БД, выполняется простой запрос:</p>
5 <p>import MySQLdb …. cursor.execute("UPDATE User SET likes=likes+1 WHERE Id=%s", (user_id,))</p>
5 <p>import MySQLdb …. cursor.execute("UPDATE User SET likes=likes+1 WHERE Id=%s", (user_id,))</p>
6 <p>Кажется, всё хорошо. Даже если тут же сделать SELECT, то можно убедиться, что у пользователя увеличилось число "лайков". Но есть нюанс: если тот же SELECT сделать из консольного клиента, то пользователь как будто и не обновлялся. А если посмотреть<strong>SHOW PROCESSLIST</strong>, то видно, что запрос в состоянии<strong>"Waiting for table metadata lock"</strong>.</p>
6 <p>Кажется, всё хорошо. Даже если тут же сделать SELECT, то можно убедиться, что у пользователя увеличилось число "лайков". Но есть нюанс: если тот же SELECT сделать из консольного клиента, то пользователь как будто и не обновлялся. А если посмотреть<strong>SHOW PROCESSLIST</strong>, то видно, что запрос в состоянии<strong>"Waiting for table metadata lock"</strong>.</p>
7 <p>А вся штука в настройках по умолчанию, о которых часто забывают. В частности, в MySQL последних версий по умолчанию движок таблицы InnoDB транзакционный. А в питонячьем MySQLdb с версии 1.2.0 опция<strong>autocommit</strong>выставлена в False.</p>
7 <p>А вся штука в настройках по умолчанию, о которых часто забывают. В частности, в MySQL последних версий по умолчанию движок таблицы InnoDB транзакционный. А в питонячьем MySQLdb с версии 1.2.0 опция<strong>autocommit</strong>выставлена в False.</p>
8 <h2>Что же получается?</h2>
8 <h2>Что же получается?</h2>
9 <p>Получается, что все ваши запросы в таком случае выполняются в рамках одной транзакции и не видны другим транзакциям до явного вызова commit (или rollback). Плюс, лочатся метаданные используемой таблицы, что отображается в processlist’е.</p>
9 <p>Получается, что все ваши запросы в таком случае выполняются в рамках одной транзакции и не видны другим транзакциям до явного вызова commit (или rollback). Плюс, лочатся метаданные используемой таблицы, что отображается в processlist’е.</p>
10 <h2>Что делать?</h2>
10 <h2>Что делать?</h2>
11 <p>Можно вызывать<strong>conn.commit()</strong>в конце транзакции, а можно сразу после установления соединения выставить<strong>conn.autocommit(True)</strong>, тогда каждый запрос будет завершаться коммитом прозрачно для вас. Ну и конечно, нужно внимательно читать документацию и changelog’и.</p>
11 <p>Можно вызывать<strong>conn.commit()</strong>в конце транзакции, а можно сразу после установления соединения выставить<strong>conn.autocommit(True)</strong>, тогда каждый запрос будет завершаться коммитом прозрачно для вас. Ну и конечно, нужно внимательно читать документацию и changelog’и.</p>
12  
12