27 февраля 2020

Как подружить Python и базы данных SQL. Подробное руководство

Пишу, перевожу и иллюстрирую IT-статьи. На proglib написал 140 материалов. Увлекаюсь Python, вебом и Data Science. Открыт к диалогу – ссылки на соцсети и мессенджеры: https://matyushkin.github.io/links/ Если понравился стиль изложения, упорядоченный список публикаций — https://github.com/matyushkin/lessons
Рассказываем, как связать Python и реляционные базы данных трёх типов: SQLite, MySQL и PostgreSQL. Зная одну библиотеку для работы с SQL, вы легко разберетесь в остальных.

Все приложения взаимодействуют с данными, чаще всего через систему управления базами данных (СУБД). Одни языки программирования поставляются с модулями для работы с СУБД, другие требуют использования сторонних пакетов. Из этого подробного руководства вы узнаете о различных библиотеках Python для работы с SQL-базами данных. Мы разработаем простое приложение для взаимодействия с БД SQLite, MySQL и PostgreSQL.

Примечание. Если вы не разбираетесь в базах данных, советуем обратить внимание на следующие публикации Библиотеки программиста: 11 типов современных баз данных, SQL за 20 минут, Подборка материалов для изучения баз данных и SQL.

Из этого пособия вы узнаете:

  • как подключиться к различным СУБД с помощью библиотек Python для работы с SQL базами данных;
  • как управлять базами данных SQLite, MySQL и PostgreSQL;
  • как выполнять запросы к базе данных внутри приложения Python;
  • как разрабатывать приложения для разных баз данных.

Чтобы получить максимальную отдачу от этого учебного пособия, необходимо знать основы Python, SQL и работы с СУБД. Вы также должны иметь возможность загружать и импортировать пакеты в Python и знать, как устанавливать и запускать серверы БД локально или удаленно.

Содержание статьи:

  1. Схема базы данных
  2. Подключение к базам данных
  3. Создание таблиц
  4. Вставка записей
  5. Извлечение записей
  6. Обновление содержания
  7. Удаление записей таблицы

В каждом разделе по три подраздела: SQLite, MySQL и PostgreSQL.

1. Схема базы данных для обучения

В этом уроке мы разработаем очень маленькую базу данных приложения для социальных сетей. База данных будет состоять из четырех таблиц:

  1. users
  2. posts
  3. comments
  4. likes

Схема базы данных показана на рисунке ниже.

Как подружить Python и базы данных SQL. Подробное руководство

Пользователи (users) и публикации (posts) будут находиться иметь тип связи один-ко-многим: одному читателю может понравиться несколько постов. Точно так же один и тот же юзер может оставлять много комментариев (comments), а один пост может иметь несколько комментариев. Таким образом, и users, и posts по отношению к comments имеют тот же тип связи. А лайки (likes) в этом плане идентичны комментариям.

2. Подключение к базам данных

Прежде чем взаимодействовать с любой базой данных через SQL-библиотеку, с ней необходимо связаться. В этом разделе мы рассмотрим, как подключиться из приложения Python к базам данных SQLite , MySQL и PostgreSQL. Рекомендуем сделать собственный .py файл для каждой из трёх баз данных.

Примечание. Для выполнения разделов о MySQL и PostgreSQL необходимо самостоятельно запустить соответствующие серверы. Для быстрого ознакомления с тем, как запустить сервер MySQL, ознакомьтесь с разделом MySQL в публикации Запуск проекта Django (англ.). Чтобы узнать, как создать базу данных в PostgreSQL, перейдите к разделу Setting Up a Database в публикации Предотвращение атак SQL-инъекций с помощью Python (англ.).

🐍🎓 Библиотека собеса по Python
Подтянуть свои знания по Python вы можете на нашем телеграм-канале «Библиотека собеса по Python»
🐍🧩 Библиотека задач по Python
Интересные задачи по Python для практики можно найти на нашем телеграм-канале «Библиотека задач по Python»

SQLite

SQLite, вероятно, является самой простой базой данных, к которой можно подключиться с помощью Python, поскольку для этого не требуется устанавливать какие-либо внешние модули. По умолчанию стандартная библиотека Python уже содержит модуль sqlite3.

Более того, SQLite база данных не требует сервера и самодостаточна, то есть просто читает и записывает данные в файл. Подключимся с помощью sqlite3 к базе данных:

        import sqlite3
from sqlite3 import Error

def create_connection(path):
    connection = None
    try:
        connection = sqlite3.connect(path)
        print("Connection to SQLite DB successful")
    except Error as e:
        print(f"The error '{e}' occurred")

    return connection
    

Вот как работает этот код:

  • Строки 1 и 2 – импорт sqlite3 и класса Error.
  • Строка 4 определяет функцию create_connection(), которая принимает путь к базе данных SQLite.
  • Строка 7 использует метод connect() и принимает в качестве параметра путь к базе данных SQLite. Если база данных в указанном месте существует, будет установлено соединение. В противном случае по указанному пути будет создана новая база данных и так же установлено соединение.
  • В строке 8 выводится состояние успешного подключения к базе данных.
  • Строка 9 перехватывает любое исключение, которое может быть получено, если методу .connect() не удастся установить соединение.
  • В строке 10 отображается сообщение об ошибке в консоли.

sqlite3.connect(path) возвращает объект connection. Этот объект может использоваться для выполнения запросов к базе данных SQLite. Следующий скрипт формирует соединение с базой данных SQLite:

        connection = create_connection("E:\\sm_app.sqlite")
    

Выполнив вышеуказанный скрипт, вы увидите, как в корневом каталоге диска E появится файл базы данных sm_app.sqlite. Конечно, вы можете изменить местоположение в соответствии с вашими интересами.

MySQL

В отличие от SQLite, в Python по умолчанию нет модуля, который можно использовать для подключения к базе данных MySQL. Для этого вам нужно установить драйвер Python для MySQL. Одним из таких драйверов является mysql-connector-python. Вы можете скачать этот модуль Python SQL с помощью pip:

        pip install mysql-connector-python
    

Обратите внимание, что MySQL – это серверная система управления базами данных. Один сервер MySQL может хранить несколько баз данных. В отличие от SQLite, где соединение равносильно порождению БД, формирование базы данных MySQL состоит из двух этапов:

  1. Установка соединения с сервером MySQL.
  2. Выполнение запроса для создания БД.

Определим функцию, которая будет подключаться к серверу MySQL и возвращать объект подключения:

        import mysql.connector
from mysql.connector import Error

def create_connection(host_name, user_name, user_password):
    connection = None
    try:
        connection = mysql.connector.connect(
            host=host_name,
            user=user_name,
            passwd=user_password
        )
        print("Connection to MySQL DB successful")
    except Error as e:
        print(f"The error '{e}' occurred")

    return connection

connection = create_connection("localhost", "root", "")
    

В приведенном выше коде мы определили новую функцию create_connection(), которая принимает три параметра:

  1. host_name
  2. user_name
  3. user_password

Модуль mysql.connector определяет метод connect(), используемый в седьмой строке для подключения к серверу MySQL. Как только соединение установлено, объект connection возвращается вызывающей функции. В последней строке функция create_connection() вызывается с именем хоста, именем пользователя и паролем.

Пока мы только установили соединение. Самой базы ещё нет. Для этого мы определим другую функцию – create_database(), которая принимает два параметра:

  1. Объект connection;
  2. query – строковый запрос о создании базу данных.

Вот как выглядит эта функция:

        def create_database(connection, query):
    cursor = connection.cursor()
    try:
        cursor.execute(query)
        print("Database created successfully")
    except Error as e:
        print(f"The error '{e}' occurred")
    

Для выполнения запросов используется объект cursor.

Создадим базу данных sm_app для нашего приложения на сервере MySQL:

        create_database_query = "CREATE DATABASE sm_app"
create_database(connection, create_database_query)
    

Теперь у нас есть база данных на сервере. Однако объект connection, возвращаемый функцией create_connection() подключен к серверу MySQL. А нам необходимо подключиться к базе данных sm_app. Для этого нужно изменить create_connection() следующим образом:

        def create_connection(host_name, user_name, user_password, db_name):
    connection = None
    try:
        connection = mysql.connector.connect(
            host=host_name,
            user=user_name,
            passwd=user_password,
            database=db_name
        )
        print("Connection to MySQL DB successful")
    except Error as e:
        print(f"The error '{e}' occurred")

    return connection
    

Функция create_connection() теперь принимает дополнительный параметр с именем db_name. Этот параметр указывает имя БД, к которой мы хотим подключиться. Имя теперь можно передать при вызове функции:

        connection = create_connection("localhost", "root", "", "sm_app")
    

Скрипт успешно вызывает create_connection() и подключается к базе данных sm_app.

PostgreSQL

Как и в случае MySQL, для PostgreSQL в стандартной библиотеке Python нет модуля для взаимодействия с базой данных. Но и для этой задачи есть решение – модуль psycopg2:

        pip install psycopg2
    

Определим функцию create_connection() для подключения к базе данных PostgreSQL:

        from psycopg2 import OperationalError

def create_connection(db_name, db_user, db_password, db_host, db_port):
    connection = None
    try:
        connection = psycopg2.connect(
            database=db_name,
            user=db_user,
            password=db_password,
            host=db_host,
            port=db_port,
        )
        print("Connection to PostgreSQL DB successful")
    except OperationalError as e:
        print(f"The error '{e}' occurred")
    return connection
    

Подключение осуществляется через интерфейс psycopg2.connect(). Далее используем написанную нами функцию:

        connection = create_connection(
    "postgres", "postgres", "abc123", "127.0.0.1", "5432"
)
    

Теперь внутри дефолтной БД postgres нужно создать базу данных sm_app. Ниже определена соответствующая функция create_database():

        def create_database(connection, query):
    connection.autocommit = True
    cursor = connection.cursor()
    try:
        cursor.execute(query)
        print("Query executed successfully")
    except OperationalError as e:
        print(f"The error '{e}' occurred")

create_database_query = "CREATE DATABASE sm_app"
create_database(connection, create_database_query)
    

Запустив вышеприведенный скрипт, мы увидим базу данных sm_app на своем сервере PostgreSQL. Подключимся к ней:

        connection = create_connection(
    "sm_app", "postgres", "abc123", "127.0.0.1", "5432"
)

    

Здесь 127.0.0.1 и 5432 это соответственно IP-адресу и порт хоста сервера.

🐍 Библиотека питониста
Больше полезных материалов вы найдете на нашем телеграм-канале «Библиотека питониста»

3. Создание таблиц

В предыдущем разделе мы увидели, как подключаться к серверам баз данных SQLite, MySQL и PostgreSQL, используя разные библиотеки Python. Мы создали базу данных sm_app на всех трех серверах БД. В данном разделе мы рассмотрим, как формировать таблицы внутри этих трех баз данных.

Как обсуждалось ранее, нам нужно получить и связать четыре таблицы:

  1. users
  2. posts
  3. comments
  4. likes

SQLite

Для выполнения запросов в SQLite используется метод cursor.execute(). В этом разделе мы определим функцию execute_query(), которая использует этот метод. Функция будет принимать объект connection и строку запроса. Далее строка запроса будет передаваться методу execute( ). В этом разделе он будет использоваться для формирования таблиц, а в следующих – мы применим его для выполнения запросов на обновление и удаление.

Примечание. Описываемый далее скрипт – часть того же файла, в котором мы описали соединение с базой данных SQLite.

Итак, начнем с определения функции execute_query():

        def execute_query(connection, query):
    cursor = connection.cursor()
    try:
        cursor.execute(query)
        connection.commit()
        print("Query executed successfully")
    except Error as e:
        print(f"The error '{e}' occurred")

    

Теперь напишем передаваемый запрос (query):

        create_users_table = """
CREATE TABLE IF NOT EXISTS users (
  id INTEGER PRIMARY KEY AUTOINCREMENT,
  name TEXT NOT NULL,
  age INTEGER,
  gender TEXT,
  nationality TEXT
);
"""

    

В запросе говорится, что нужно создать таблицу users со следующими пятью столбцами:

  1. id
  2. name
  3. age
  4. gender
  5. nationality

Наконец, чтобы появилась таблица, вызываем execute_query(). Передаём объект connection, который мы описали в предыдущем разделе, вместе с только что подготовленной строкой запроса create_users_table:

        execute_query(connection, create_users_table)  

    

Следующий запрос используется для создания таблицы posts:

        create_posts_table = """
CREATE TABLE IF NOT EXISTS posts(
  id INTEGER PRIMARY KEY AUTOINCREMENT, 
  title TEXT NOT NULL, 
  description TEXT NOT NULL, 
  user_id INTEGER NOT NULL, 
  FOREIGN KEY (user_id) REFERENCES users (id)
);
"""

    

Поскольку между users и posts имеет место отношение один-ко-многим, в таблице появляется ключ user_id, который ссылается на столбец id в таблице users. Выполняем следующий скрипт для построения таблицы posts:

        execute_query(connection, create_posts_table)

    

Наконец, формируем следующим скриптом таблицы comments и likes:

        create_comments_table = """
CREATE TABLE IF NOT EXISTS comments (
  id INTEGER PRIMARY KEY AUTOINCREMENT, 
  text TEXT NOT NULL, 
  user_id INTEGER NOT NULL, 
  post_id INTEGER NOT NULL, 
  FOREIGN KEY (user_id) REFERENCES users (id) FOREIGN KEY (post_id) REFERENCES posts (id)
);
"""

create_likes_table = """
CREATE TABLE IF NOT EXISTS likes (
  id INTEGER PRIMARY KEY AUTOINCREMENT, 
  user_id INTEGER NOT NULL, 
  post_id integer NOT NULL, 
  FOREIGN KEY (user_id) REFERENCES users (id) FOREIGN KEY (post_id) REFERENCES posts (id)
);
"""

execute_query(connection, create_comments_table)  
execute_query(connection, create_likes_table)            

    

Вы могли заметить, что создание таблиц в SQLite очень похоже на использование чистого SQL. Все, что вам нужно сделать, это сохранить запрос в строковой переменной и затем передать эту переменную cursor.execute().

MySQL

Так же, как с SQLite, чтобы создать таблицу в MySQL, нужно передать запрос в cursor.execute(). Создадим новый вариант функции execute_query():

        def execute_query(connection, query):
     cursor = connection.cursor()
     try:
         cursor.execute(query)
         connection.commit()
         print("Query executed successfully")
     except Error as e:
         print(f"The error '{e}' occurred")

    

Описываем таблицу users:

        create_users_table = """
CREATE TABLE IF NOT EXISTS users (
  id INT AUTO_INCREMENT, 
  name TEXT NOT NULL, 
  age INT, 
  gender TEXT, 
  nationality TEXT, 
  PRIMARY KEY (id)
) ENGINE = InnoDB
"""

execute_query(connection, create_users_table)

    

Запрос для реализации отношения внешнего ключа в MySQL немного отличается от SQLite. Более того, MySQL использует ключевое слово AUTO_INCREMENT для указания столбцов, значения которых автоматически увеличиваются при вставке новых записей.

Следующий скрипт составит таблицу posts, содержащую внешний ключ user_id, который ссылается на id столбца таблицы users:

        create_posts_table = """
CREATE TABLE IF NOT EXISTS posts (
  id INT AUTO_INCREMENT, 
  title TEXT NOT NULL, 
  description TEXT NOT NULL, 
  user_id INTEGER NOT NULL, 
  FOREIGN KEY fk_user_id (user_id) REFERENCES users(id), 
  PRIMARY KEY (id)
) ENGINE = InnoDB
"""

execute_query(connection, create_posts_table)

    

Аналогично для создания таблиц comments и likes, передаём соответствующие CREATE-запросы функции execute_query().

PostgreSQL

Применение библиотеки psycopg2 в execute_query() также подразумевает работу с cursor:

        def execute_query(connection, query):
    connection.autocommit = True
    cursor = connection.cursor()
    try:
        cursor.execute(query)
        print("Query executed successfully")
    except OperationalError as e:
        print(f"The error '{e}' occurred")

    

Мы можем использовать эту функцию для организации таблиц, вставки, изменения и удаления записей в вашей базе данных PostgreSQL.

Создадим внутри базы данных sm_app таблицу users :

        create_users_table = """
CREATE TABLE IF NOT EXISTS users (
  id SERIAL PRIMARY KEY,
  name TEXT NOT NULL, 
  age INTEGER,
  gender TEXT,
  nationality TEXT
)
"""

execute_query(connection, create_users_table)

    

Запрос на создание таблицы users в PostgreSQL немного отличается от SQLite и MySQL. Здесь для указания столбцов с автоматическим инкрементом используется ключевое слово SERIAL. Кроме того, отличается способ указания ссылок на внешние ключи:

        create_posts_table = """
CREATE TABLE IF NOT EXISTS posts (
  id SERIAL PRIMARY KEY, 
  title TEXT NOT NULL, 
  description TEXT NOT NULL, 
  user_id INTEGER REFERENCES users(id)
)
"""

execute_query(connection, create_posts_table)

    

4. Вставка записей

В предыдущем разделе мы разобрали, как развертывать таблицы в базах данных SQLite, MySQL и PostgreSQL с использованием различных модулей Python. В этом разделе узнаем, как вставлять записи.

SQLite

Чтобы вставить записи в базу данных SQLite, мы можем использовать ту же execute_query() функцию, что и для создания таблиц. Для этого сначала нужно сохранить в виде строки запрос INSERT INTO. Затем нужно передать объект connection и строковый запрос в execute_query(). Вставим для примера пять записей в таблицу users:

        create_users = """
INSERT INTO
  users (name, age, gender, nationality)
VALUES
  ('James', 25, 'male', 'USA'),
  ('Leila', 32, 'female', 'France'),
  ('Brigitte', 35, 'female', 'England'),
  ('Mike', 40, 'male', 'Denmark'),
  ('Elizabeth', 21, 'female', 'Canada');
"""

execute_query(connection, create_users)   

    

Поскольку мы установили автоинкремент для столбца id, нам не нужно указывать его дополнительно. Таблица users будет автоматически заполнена пятью записями со значениями id от 1 до 5.

Вставим в таблицу posts шесть записей:

        create_posts = """
INSERT INTO
  posts (title, description, user_id)
VALUES
  ("Happy", "I am feeling very happy today", 1),
  ("Hot Weather", "The weather is very hot today", 2),
  ("Help", "I need some help with my work", 2),
  ("Great News", "I am getting married", 1),
  ("Interesting Game", "It was a fantastic game of tennis", 5),
  ("Party", "Anyone up for a late-night party today?", 3);
"""

execute_query(connection, create_posts)  

    

Важно отметить, что столбец user_id таблицы posts является внешним ключом, который ссылается на столбец таблицы users. Это означает, что столбец user_id должен содержать значение, которое уже существует в столбце id таблицы users. Если его не существует, мы получим сообщение об ошибке.

Следующий скрипт вставляет записи в таблицы comments и likes:

        create_comments = """
INSERT INTO
  comments (text, user_id, post_id)
VALUES
  ('Count me in', 1, 6),
  ('What sort of help?', 5, 3),
  ('Congrats buddy', 2, 4),
  ('I was rooting for Nadal though', 4, 5),
  ('Help with your thesis?', 2, 3),
  ('Many congratulations', 5, 4);
"""

create_likes = """
INSERT INTO
  likes (user_id, post_id)
VALUES
  (1, 6),
  (2, 3),
  (1, 5),
  (5, 4),
  (2, 4),
  (4, 2),
  (3, 6);
"""

execute_query(connection, create_comments)
execute_query(connection, create_likes)  

    

MySQL

Есть два способа вставить записи в базы данных MySQL из приложения Python. Первый подход похож на SQLite. Можно сохранить запрос INSERT INTO в строке, а затем использовать для вставки записей cursor.execute().

Ранее мы определили функцию-оболочку execute_query(), которую использовали для вставки записей. Мы можем использовать ту же функцию:

        create_users = """
INSERT INTO
  `users` (`name`, `age`, `gender`, `nationality`)
VALUES
  ('James', 25, 'male', 'USA'),
  ('Leila', 32, 'female', 'France'),
  ('Brigitte', 35, 'female', 'England'),
  ('Mike', 40, 'male', 'Denmark'),
  ('Elizabeth', 21, 'female', 'Canada');
"""

execute_query(connection, create_users)  

    

Второй подход использует метод cursor.executemany(), который принимает два параметра:

  1. Строка query, содержащая заполнители для вставляемых записей.
  2. Список записей, которые мы хотим вставить.

Посмотрите на следующий пример, который вставляет две записи в таблицу likes:

        sql = "INSERT INTO likes ( user_id, post_id ) VALUES ( %s, %s )"
val = [(4, 5), (3, 4)]

cursor = connection.cursor()
cursor.executemany(sql, val)
connection.commit()

    

Какой подход выбрать – зависит от вас. Если вы не очень хорошо знакомы с SQL, проще использовать метод курсора executemany().

PostgreSQL

В предыдущем подразделе мы познакомились с двумя подходами для вставки записей в таблицы баз данных MySQL. В psycopg2 используется второй подход: мы передаем SQL-запрос с заполнителями и списком записей методу execute(). Каждая запись в списке должна являться кортежем, значения которого соответствуют значениям столбца в таблице БД. Вот как мы можем вставить пользовательские записи в таблицу users:

        users = [
    ("James", 25, "male", "USA"),
    ("Leila", 32, "female", "France"),
    ("Brigitte", 35, "female", "England"),
    ("Mike", 40, "male", "Denmark"),
    ("Elizabeth", 21, "female", "Canada"),
]

user_records = ", ".join(["%s"] * len(users))

insert_query = (
    f"INSERT INTO users (name, age, gender, nationality) VALUES {user_records}"
)

connection.autocommit = True
cursor = connection.cursor()
cursor.execute(insert_query, users)

    

Список users содержит пять пользовательских записей в виде кортежей. Затем мы создаём строку с пятью элементами-заполнителями (%s), соответствующими пяти пользовательским записям. Строка-заполнитель объединяется с запросом, который вставляет записи в таблицу users. Наконец, строка запроса и пользовательские записи передаются в метод execute().

Следующий скрипт вставляет записи в таблицу posts:

        posts = [
    ("Happy", "I am feeling very happy today", 1),
    ("Hot Weather", "The weather is very hot today", 2),
    ("Help", "I need some help with my work", 2),
    ("Great News", "I am getting married", 1),
    ("Interesting Game", "It was a fantastic game of tennis", 5),
    ("Party", "Anyone up for a late-night party today?", 3),
]

post_records = ", ".join(["%s"] * len(posts))

insert_query = (
    f"INSERT INTO posts (title, description, user_id) VALUES {post_records}"
)

connection.autocommit = True
cursor = connection.cursor()
cursor.execute(insert_query, posts)

    

По той же методике можно вставить записи в таблицы comments и likes.

5. Извлечение данных из записей

SQLite

Чтобы выбрать записи в SQLite, можно снова использовать cursor.execute(). Однако после этого потребуется вызвать метод курсора fetchall(). Этот метод возвращает список кортежей, где каждый кортеж сопоставлен с соответствующей строкой в ​​извлеченных записях. Чтобы упростить процесс, напишем функцию execute_read_query():

        def execute_read_query(connection, query):
    cursor = connection.cursor()
    result = None
    try:
        cursor.execute(query)
        result = cursor.fetchall()
        return result
    except Error as e:
        print(f"The error '{e}' occurred")

    

Эта функция принимает объект connection и SELECT-запрос, а возвращает выбранную запись.

SELECT

Давайте выберем все записи из таблицы users:

        select_users = "SELECT * from users"
users = execute_read_query(connection, select_users)

for user in users:
    print(user)

    

В приведенном выше скрипте запрос SELECT забирает всех пользователей из таблицы users. Результат передается в написанную нами функцию execute_read_query(), возвращающую все записи из таблицы users.

Примечание. Не рекомендуется использовать SELECT * для больших таблиц, так как это может привести к большому числу операций ввода-вывода, которые увеличивают сетевой трафик.

Результат вышеприведенного запроса выглядит следующим образом:

        (1, 'James', 25, 'male', 'USA')
(2, 'Leila', 32, 'female', 'France')
(3, 'Brigitte', 35, 'female', 'England')
(4, 'Mike', 40, 'male', 'Denmark')
(5, 'Elizabeth', 21, 'female', 'Canada')

    

Таким же образом вы можете извлечь все записи из таблицы posts:

        select_posts = "SELECT * FROM posts"
posts = execute_read_query(connection, select_posts)

for post in posts:
    print(post)

    

Вывод выглядит так:

        (1, 'Happy', 'I am feeling very happy today', 1)
(2, 'Hot Weather', 'The weather is very hot today', 2)
(3, 'Help', 'I need some help with my work', 2)
(4, 'Great News', 'I am getting married', 1)
(5, 'Interesting Game', 'It was a fantastic game of tennis', 5)
(6, 'Party', 'Anyone up for a late-night party today?', 3)

    

JOIN

Вы также можете выполнять более сложные запросы, включающие операции типа JOIN для извлечения данных из двух связанных таблиц. Например, следующий скрипт возвращает идентификаторы и имена пользователей, а также описание сообщений, опубликованных этими пользователями:

        select_users_posts = """
SELECT
  users.id,
  users.name,
  posts.description
FROM
  posts
  INNER JOIN users ON users.id = posts.user_id
"""

users_posts = execute_read_query(connection, select_users_posts)

for users_post in users_posts:
    print(users_post)

    

Вывод данных:

        (1, 'James', 'I am feeling very happy today')
(2, 'Leila', 'The weather is very hot today')
(2, 'Leila', 'I need some help with my work')
(1, 'James', 'I am getting married')
(5, 'Elizabeth', 'It was a fantastic game of tennis')
(3, 'Brigitte', 'Anyone up for a late night party today?')

    

Следующий скрипт возвращает все сообщения вместе с комментариями к сообщениям и именами пользователей, которые разместили комментарии:

        select_posts_comments_users = """
SELECT
  posts.description as post,
  text as comment,
  name
FROM
  posts
  INNER JOIN comments ON posts.id = comments.post_id
  INNER JOIN users ON users.id = comments.user_id
"""

posts_comments_users = execute_read_query(
    connection, select_posts_comments_users
)

for posts_comments_user in posts_comments_users:
    print(posts_comments_user)

    

Вывод выглядит так:

        ('Anyone up for a late night party today?', 'Count me in', 'James')
('I need some help with my work', 'What sort of help?', 'Elizabeth')
('I am getting married', 'Congrats buddy', 'Leila')
('It was a fantastic game of tennis', 'I was rooting for Nadal though', 'Mike')
('I need some help with my work', 'Help with your thesis?', 'Leila')
('I am getting married', 'Many congratulations', 'Elizabeth')

    

Из вывода понятно, что имена столбцов не были возвращены методом fetchall(). Чтобы вернуть имена столбцов, нужно забрать атрибут description объекта cursor. Например, следующий список возвращает все имена столбцов для вышеуказанного запроса:

        cursor = connection.cursor()
cursor.execute(select_posts_comments_users)
cursor.fetchall()

column_names = [description[0] for description in cursor.description]
print(column_names)

    

Вывод выглядит так:

        ['post', 'comment', 'name']

    

WHERE

Теперь мы выполним SELECT-запрос, который возвращает текст поста и общее количество лайков, им полученных:

        select_post_likes = """
SELECT
  description as Post,
  COUNT(likes.id) as Likes
FROM
  likes,
  posts
WHERE
  posts.id = likes.post_id
GROUP BY
  likes.post_id
"""

post_likes = execute_read_query(connection, select_post_likes)

for post_like in post_likes:
    print(post_like)

    

Вывод следующий:

        ('The weather is very hot today', 1)
('I need some help with my work', 1)
('I am getting married', 2)
('It was a fantastic game of tennis', 1)
('Anyone up for a late night party today?', 2)

    

То есть используя запрос WHERE, вы можете возвращать более конкретные результаты.

MySQL

Процесс выбора записей в MySQL абсолютно идентичен процессу выбора записей в SQLite:

        def execute_read_query(connection, query):
    cursor = connection.cursor()
    result = None
    try:
        cursor.execute(query)
        result = cursor.fetchall()
        return result
    except Error as e:
        print(f"The error '{e}' occurred")

    

Теперь выберем все записи из таблицы users:

        select_users = "SELECT * FROM users"
users = execute_read_query(connection, select_users)

for user in users:
    print(user)

    

Вывод будет похож на то, что мы видели с SQLite.

PostgreSQL

Процесс выбора записей из таблицы PostgreSQL с помощью модуля psycopg2 тоже похож на SQLite и MySQL. Снова используем cursor.execute(), затем метод fetchall() для выбора записей из таблицы. Следующий скрипт выбирает все записи из таблицы users:

        def execute_read_query(connection, query):
    cursor = connection.cursor()
    result = None
    try:
        cursor.execute(query)
        result = cursor.fetchall()
        return result
    except OperationalError as e:
        print(f"The error '{e}' occurred")

select_users = "SELECT * FROM users"
users = execute_read_query(connection, select_users)

for user in users:
    print(user)

    

Опять же, результат будет похож на то, что мы видели раньше.

6. Обновление записей таблицы

SQLite

Обновление записей в SQLite выглядит довольно просто. Снова можно применить execute_query(). В качестве примера обновим текст поста с id равным 2. Сначала создадим описание для SELECT:

        select_post_description = "SELECT description FROM posts WHERE id = 2"

post_description = execute_read_query(connection, select_post_description)

for description in post_description:
    print(description)

    

Увидим следующий вывод:

        ('The weather is very hot today',)

    

Следующий скрипт обновит описание:

        update_post_description = """
UPDATE
  posts
SET
  description = "The weather has become pleasant now"
WHERE
  id = 2
"""

execute_query(connection, update_post_description)

    

Теперь, если мы выполним SELECT-запрос еще раз, увидим следующий результат:

        ('The weather has become pleasant now',)

    

То есть запись была обновлена.

MySQL

Процесс обновления записей в MySQL с помощью модуля mysql-connector-python является точной копией модуля sqlite3:

        update_post_description = """
UPDATE
  posts
SET
  description = "The weather has become pleasant now"
WHERE
  id = 2
"""

execute_query(connection,  update_post_description)

    

PostgreSQL

Запрос на обновление PostgreSQL аналогичен SQLite и MySQL.

7. Удаление записей таблицы

SQLite

В качестве примера удалим комментарий с id равным 5:

        delete_comment = "DELETE FROM comments WHERE id = 5"
execute_query(connection, delete_comment)

    

Теперь, если мы извлечем все записи из таблицы comments, то увидим, что пятый комментарий был удален. Процесс удаления в MySQL и PostgreSQL идентичен SQLite:

Заключение

В этом руководстве мы разобрались, как применять три распространенные библиотеки Python для работы с реляционными базами данных. Научившись работать с одним из модулей sqlite3, mysql-connector-python и psycopg2, вы легко сможете перенести свои знания на другие модули и оперировать любой из баз данных SQLite, MySQL и PostgreSQL.

Однако это лишь вершина айсберга! Существуют также библиотеки для работы с SQL и объектно-реляционными отображениями, такие как SQLAlchemy и Django ORM, которые автоматизируют задачи взаимодействия Python с базами данных.

Если вам интересна тематика работы с базами данных с помощью Python, напишите об этом в комментариях – мы подготовим дополнительные материалы.

Больше полезной информации вы найдете на наших телеграм-каналах «Библиотека питониста» и «Библиотека data scientist’а».

Комментарии

 
 
14 января 2024

rom sqlalchemy import create_engine, Column, Integer, TEXT, VARCHAR, DATE, DATETIME, ForeignKey, LargeBinary from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import relationship, sessionmaker from datetime import datetime import hashlib

Base = declarative_base()

engine = create_engine('mysql+mysqlconnector://user31:42107@192.168.15.95:3306/user31', echo=True)

class Users(Base): tablename = 'Polzovatel_avtorizacia'

iD_user = Column('Id_polzovatela', Integer, primary_key=True)
login = Column('Login', VARCHAR(100))
password = Column('Parol', VARCHAR(100))

Base.metadata.create_all(engine)

session = sessionmaker(bind=engine) s = session()

def log_in(login, password): try: hash_password = hashlib.md5(bytes(password, 'UTF-8')) log = s.query(Users).where(Users.login == login).one() return hash_password.hexdigest() == log.password except: return False

def login_verification(login): try: log = s.query(Users).where(Users.login == login).all() return False except: return True

def account_dannie_veridication(login, password, password_ver, message): if login == '' or password == '': message(None, 'Ошибка', 'Введите данные') return

if password != password_ver:
    message(None, 'Ошибка', 'Введены неверные данные')
    return

if login_verification(login):
    message(None, 'Ошибка', 'Такой логин уже существует')
    return

return True

def sign_up(login, password): hash_password = hashlib.md5(bytes(password, 'UTF-8')) profil = Users( login=login, password=hash_password.hexdigest() ) s.add(profil) s.commit()

class Zaiavka_Na_Remont(Base): tablename = 'Zaiavka_na_remont' Nomer_zaiavki = Column("Nomer_zaiavki", Integer, primary_key=True) ispolnitel = Column("Ispolnitel", Integer, ForeignKey('Ispolnitel.ID_ispolnitela')) Klient = Column("Klient", Integer, ForeignKey('Klienti.Id_klienta')) data_nachala_raboti = Column("Data_nachala_raboti", DATE) data_oconchania_raboti = Column("Data_oconchania_raboti", DATE) Oborydovanie = Column("Oborydovanie", VARCHAR(100)) Tip_Neispravnosti = Column("Tip_neispravnosti", VARCHAR(100)) Opisanie_Problemi = Column("Opisanie_problemi", VARCHAR(100)) StatysZaiavki = Column("Statys_zaiavki", VARCHAR(100)) Data_Obrashenia = Column("Data_obrashenia", DATE)

Base.metadata.create_all(engine)

session = sessionmaker(bind=engine) s = session()

def dannie_zaiavki(Nomer_zaiavki, ispolnitel, Klient, data_nachala_raboti, data_oconchania_raboti, Oborydovanie, Tip_Neispravnosti, Opisanie_Problemi, StatysZaiavki, Data_Obrashenia, message): if Nomer_zaiavki == '' or ispolnitel == '' or Klient == '' or data_nachala_raboti == '' or data_oconchania_raboti == '' or Oborydovanie == '' or Tip_Neispravnosti == '' or Opisanie_Problemi == '' or StatysZaiavki == '' or Data_Obrashenia == '': message(None, 'Ошибка', 'Введите данные') return

return True

def dannie(Nomer_zaiavki, ispolnitel, Klient, data_nachala_raboti, data_oconchania_raboti, Oborydovanie, Tip_Neispravnosti, Opisanie_Problemi, StatysZaiavki, Data_Obrashenia): profil = Zaiavka_Na_Remont( Nomer_zaiavki=Nomer_zaiavki, ispolnitel=ispolnitel, Klient=Klient, data_nachala_raboti=data_nachala_raboti, data_oconchania_raboti=data_oconchania_raboti, Oborydovanie=Oborydovanie, Tip_Neispravnosti=Tip_Neispravnosti, Opisanie_Problemi=Opisanie_Problemi, StatysZaiavki=StatysZaiavki, Data_Obrashenia=Data_Obrashenia, ) s.add(profil) s.commit()

class Ispolnitely(Base): tablename = 'Ispolnitel'

Id_ispolnitela = Column('ID_ispolnitela', Integer, primary_key=True)
Familia = Column('familia', VARCHAR(100))
Ima = Column('ima', VARCHAR(100))
Otchestvo = Column('otchestvo', VARCHAR(100))
Telefon = Column('Phone', VARCHAR(100))
Rabochii_grafic = Column('Rabochii_graphic', VARCHAR(100))

Base.metadata.create_all(engine)

session = sessionmaker(bind=engine) s = session()

def dannie_ispolnitela(Id_ispolnitela, Familia, Ima, Otchestvo, Telefon, Rabochii_grafic, message): if Id_ispolnitela == '' or Familia == '' or Ima == '' or Otchestvo == '' or Telefon == '' or Rabochii_grafic == '': message(None, 'Ошибка', 'Введите данные') return

return True

def dannie_ispolnitelei(Id_ispolnitela, Familia, Ima, Otchestvo, Telefon, Rabochii_grafic): profil = Ispolnitely( Id_ispolnitela=Id_ispolnitela, Familia=Familia, Ima=Ima, Otchestvo=Otchestvo, Telefon=Telefon, Rabochii_grafic=Rabochii_grafic, ) s.add(profil) s.commit()

class Klienty(Base): tablename = 'Klienti'

Id_klienta = Column('Id_klienta', Integer, primary_key=True)
Familia_klienti = Column('Familia', VARCHAR(100))
Ima_klienti = Column('Ima', VARCHAR(100))
Otchestvo_klienti = Column('Otchestvo', VARCHAR(100))
Telefon_klienti = Column('Telephon', VARCHAR(100))
Data_obrashenia_klienti = Column('Data_obrashenia', DATE)

Base.metadata.create_all(engine)

session = sessionmaker(bind=engine) s = session()

def dannie_klienta(Id_klienta, Familia_klienti, Ima_klienti, Otchestvo_klienti, Telefon_klienti, Data_obrashenia_klienti, message): if Id_klienta == '' or Familia_klienti == '' or Ima_klienti == '' or Otchestvo_klienti == '' or Telefon_klienti == '' or Data_obrashenia_klienti == '': message(None, 'Ошибка', 'Введите данные') return

return True

def dannie_klienty(Id_klienta, Familia_klienti, Ima_klienti, Otchestvo_klienti, Telefon_klienti, Data_obrashenia_klienti): profil = Klienty( Id_klienta=Id_klienta, Familia_klienti=Familia_klienti, Ima_klienti=Ima_klienti, Otchestvo_klienti=Otchestvo_klienti, Telefon_klienti=Telefon_klienti, Data_obrashenia_klienti=Data_obrashenia_klienti ) s.add(profil) s.commit()

29 октября 2023

Для тех, у кого возникнет проблема с установкой psycopg2 на Linux - Xubuntu/Ubuntu (Сам столкнулся с подобным) :

sudo apt-get update

sudo apt-get install libpq-dev python3-dev

sudo apt-get install python3-pip

sudo pip install psycopg2

24 февраля 2023

Добрый день. Благодарю за познаватлеьный материал. Было бы полезно осветить вопрос необходимости закрытия курсора и коннекта. Заранее благодарю.

21 января 2023

Очень интересная статья, половина курса SQL доступно расписана в кратком варианте, спасибо.

25 декабря 2022

Спасибо за статью, очень познавательно.

При попытке создать БД в постгресе выдает такую ошибку: AttributeError: 'NoneType' object has no attribute 'autocommit'

Скопировал все без изменений. В чем дело - понять не могу. По идее, все должно работать. Прошу разъяснить в чем дело. п.с. может я по кусочкам код криво скопировал?

03 февраля 2022

При попытке INSERT не получается это сделать, т.к. не воспринимается регистр букв, а имя солонки имеет разный регистр, как учесть регистр, если он не учитывается автоматически? columns = 'id, Number, OS, test, Test_new' insert_query = (f"INSERT INTO detection( {columns} ) VALUES {parameter_record}")

psycopg2.errors.UndefinedColumn: column "number" of relation "detection" does not exist LINE 1: id, Number, OS, test, Test_new

03 февраля 2022

В PostgreSQL все идентификаторы по умолчанию ставятся в нижний регистр. Исключение — если имя столбца было создано с использованием двойных кавычек. Тогда оно сохраняет все особенности регистра. В этом случае при вызове столбца его название тоже нужно передавать в двойных кавычках, то есть "Number".

Как в качестве аргументов при добавлении записей использовать список? Имею ввиду следующее: f = open('rows.csv', 'r') rows = [] for row in f: rows.append(row)

def create_simptom(): """ INSERT INTO simptoms (name) VALUES (?) """

execute_query(connection, create_simptom(rows))

01 декабря 2020

Здесь описывается подход с использованием стандартного SQL-синтаксиса для разных баз данных. То есть сам запрос это просто строка, а значит, для вашей задачи можно просто конкатенировать строки или повторять запрос для каждой строки необходимое количество раз. Но это если задача на один раз.

Если подобных задач несколько и требуется решение, не зависящее от типа базы данных, рекомендую статью по SQLAlchemy.

Для представления кода в форматированном виде в комментариях, пожалуйста, используйте стандартную разметку Markdown: тройные обратные кавычки в начале и в конце блока программного кода.

11 октября 2020

Хотелось бы иметь полный код примера в файле для sqlite, который можно запустить и он будет работать сразу, а то копируя кусочки, не получается запустить. Выдает ошибку.

import sqlite3 from sqlite3 import Error

def create_connection(path): connection = None try: connection = sqlite3.connect(path) print("Connection to SQLite DB successful") except Error as e: print(f"The error '{e}' occurred")

return connection

connection = create_connection("sm_app.sqlite")

def execute_query(connection, query): cursor = connection.cursor() try: cursor.execute(query) connection.commit() print("Query executed successfully") except Error as e: print(f"The error '{e}' occurred")

create_users_table = """ CREATE TABLE IF NOT EXISTS users ( id INTEGER PRIMARY KEY AUTOINCREMENT, name TEXT NOT NULL, age INTEGER, gender TEXT, nationality TEXT ); """

execute_query(connection, create_users_table)

create_posts_table = """ CREATE TABLE IF NOT EXISTS posts( id INTEGER PRIMARY KEY AUTOINCREMENT, title TEXT NOT NULL, description TEXT NOT NULL, user_id INTEGER NOT NULL, FOREIGN KEY (user_id) REFERENCES users (id) ); """

execute_query(connection, create_posts_table)

FOREIGN KEY (user_id) REFERENCES users (id) FOREIGN KEY (post_id) REFERENCES posts (id) ); """

create_likes_table = """ CREATE TABLE IF NOT EXISTS likes ( id INTEGER PRIMARY KEY AUTOINCREMENT, user_id INTEGER NOT NULL, post_id integer NOT NULL, FOREIGN KEY (user_id) REFERENCES users (id) FOREIGN KEY (post_id) REFERENCES posts (id) ); """

execute_query(connection, create_comments_table)
execute_query(connection, create_likes_table)

11 октября 2020

А какую именно ошибку выдает? Для представления кода в форматированном виде в комментариях используйте стандартную разметку Markdown: тройные обратные кавычки в начале и в конце блока программного кода.

14 января 2024

from sqlalchemy import create_engine, Column, Integer, TEXT, VARCHAR, DATE, DATETIME, ForeignKey, LargeBinary from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import relationship, sessionmaker from datetime import * import hashlib

Base = declarative_base()

engine = create_engine( 'mysql+mysqlconnector://root:8885551198:3306/tweeson', echo=True)

class Users(Base): tablename = 'users'

iD_user = Column('kod_user', Integer, primary_key=True)
login = Column('login', VARCHAR(45))
password = Column('password', VARCHAR(45))

Base.metadata.create_all(engine)

session = sessionmaker(bind=engine) s = session()

def log_in(login, password): try: hash_password = hashlib.md5(bytes(password, 'UTF-8')) log = s.query(Users).where(Users.login == login).one() return hash_password.hexdigest() == log.password except: return False

def login_verification(login): try: log = s.query(Users).where(Users.login == login).all() return False except: return True

def account_data_veridication(login, password, password_ver, message): if login == '' or password == '': message(None, 'Ошибка', 'Введите данные') return

if password != password_ver:
    message(None, 'Ошибка', 'Введены неверные данные')
    return

if login_verification(login):
    message(None, 'Ошибка', 'Такой логин уже существует')
    return

return True

def sign_up(login, password): hash_password = hashlib.md5(bytes(password, 'UTF-8')) profil = Users( login=login, password=hash_password.hexdigest() ) s.add(profil) s.commit()

17 августа 2020

С SQLite одна беда - проблема с изменением уже имеющейся таблицы. Нельзя как в MySQL удалить столбец из таблицы.

11 октября 2020

Поэтому в публикации кроме простейшей SQLite рассматриваются MySQL и PostgreSQL

ВАКАНСИИ

Добавить вакансию
Разработчик C++
Москва, по итогам собеседования

ЛУЧШИЕ СТАТЬИ ПО ТЕМЕ

Подпишись

на push-уведомления