top of page

Relational Databases: An Essential Guide for Data Scientists



I. Introduction to Relational Databases

Relational databases are an integral part of data management in various industries. Let's start by understanding their fundamentals.


A. Definition and Purpose of Relational Databases

A relational database is a type of database that organizes data into one or more tables, where data types may be related to each other; these relations help structure the data. They are built using a model of data that allows the user to manipulate the data in a relational manner. The relational model enables querying of the database to combine data from different tables in meaningful ways.


Imagine a library system. It's like having separate 'tables' for 'Books', 'Authors', 'Readers', and 'Borrowed Books'. The 'Books' table might contain data about each book's title, author, and genre. The 'Authors' table might hold information about each author's name, birthplace, and books written. These tables can be connected through relations such as the book title or the author's name. These relations make it easier to find out, for example, which books by a certain author are currently borrowed.


B. Brief History of the Relational Model

The concept of the relational model was first proposed by E.F. Codd in 1970 while he was at IBM. The beauty of the relational model was its simplicity, which was in stark contrast to the complex network and hierarchical database models of the time. This model was quickly adopted and forms the basis of most modern database systems.


II. Example of a Relational Database

Now, let's understand the structure of a relational database through an example.


A. Description of a Synthetic Database and its Structure

Consider a simplified database for an online store, which includes 'Customers', 'Orders', 'Products', and 'Employees' tables. Each of these tables holds specific information. The 'Customers' table stores data about customers, 'Orders' contains all orders placed, 'Products' has information about the products, and 'Employees' keep the data about employees.


Each of these tables is related to one or more other tables, hence creating a 'relational database'. For instance, 'Orders' table might relate to 'Customers', 'Products', and 'Employees' tables.


B. Explanation of Tables within a Database


1. Introduction to the Concept of Tables

Tables are fundamental building blocks of relational databases. A table is a collection of related data entries and it consists of columns and rows. Each column corresponds to a certain attribute, while each row represents a single record.


Think of a table as a spreadsheet. Each column represents a specific piece of information (like 'Name', 'Email', 'Date of Purchase'), and each row corresponds to a single entry or person in this case.


2. Description of an Example Table ('Orders')

Let's explore the 'Orders' table from our online store database. This table might include the following columns: 'OrderID', 'CustomerID', 'EmployeeID', 'OrderDate', and 'ProductID'. Each row in this table represents a single order placed by a customer.


CREATE TABLE Orders (
    OrderID int,
    CustomerID int,
    EmployeeID int,
    OrderDate date,
    ProductID int
);

This SQL statement creates an 'Orders' table with the columns specified.


3. Explanation of a Row or Record in a Table

A row, also known as a record or tuple, is a single, implicitly structured data item in a table. In the context of our 'Orders' table, a row could be:

OrderIDCustomerIDEmployeeIDOrderDateProductID1101122023-07-30567

This row indicates that there was an order (OrderID: 1) placed by customer 101, processed by employee 12, on the date 2023-07-30 for the product 567.


4. Significance of the Primary Key in a Table

A primary key is a unique identifier for a row in a table. No two rows in a table can have the same primary key value. In the 'Orders' table, 'OrderID' could serve as the primary key.

Primary keys are like unique barcodes for every product in a store. No two products (even if they are of the same type) have the exact same barcode.

Please let me know if you'd like to proceed with the next part of this tutorial.


III. Inter-Table Relationships in a Relational Database

Let's dive into the concept of table relationships within a relational database, which is central to the relational model.


A. Explanation of How Tables are Linked

In a relational database, tables are linked using keys. A key is a specific field that serves as a unique identifier for rows in a table. This key can be used in another table to refer to specific rows. The key used to link the tables is the primary key from one table, which is used as a foreign key in the second table. A foreign key is a column or a set of columns in one table, that is used to establish a link between the data in two tables.


Imagine having your unique student ID in a university. This ID not only identifies you in the 'Students' table, but is also used in the 'Courses' table to identify all the courses you are enrolled in.


B. Examples of Table Linking Using 'CustomerID' and 'EmployeeID'

Consider our 'Orders' table. The 'CustomerID' in the 'Orders' table is a foreign key referencing the 'CustomerID' primary key in the 'Customers' table. This means that for each order, we can know which customer placed it. Similarly, 'EmployeeID' references the employee who processed the order.

SELECT Customers.CustomerName, Orders.OrderID
FROM Customers
INNER JOIN Orders ON Customers.CustomerID = Orders.CustomerID
ORDER BY Customers.CustomerName;

This SQL statement will return a list of customer names along with the orders they placed. The INNER JOIN keyword selects records that have matching values in both tables.


C. Benefits of Table Linking in Terms of Space Saving

Table linking allows us to avoid data duplication. By referencing data in different tables through keys, we maintain a single source of truth for each piece of information. For instance, we don't need to store all customer information in the 'Orders' table; we simply reference it via the 'CustomerID'.


This is like storing all student data (like name, date of birth, etc.) in a 'Students' table, and in the 'Courses' table we only store the 'studentID' for students enrolled, rather than duplicating all student data. This is space-efficient and avoids potential inconsistencies.


IV. Theoretical Background of Relational Databases

Having understood the practical aspects of relational databases, let's delve into their theoretical foundation.


A. Brief on the Contribution of "Ted" Codd to Relational Databases

Edgar F. Codd, also known as Ted Codd, was an English computer scientist who, while at IBM, invented the relational model for database management. His model set the standard for how data is organized and accessed, a standard that is still widely followed today.


B. Overview of Codd's 12 Rules or Commandments

Codd proposed 12 rules, often referred to as Codd's 12 commandments, to define what qualifies as a relational database. Some key rules include:

  1. Rule 0: A relational database must manage its stored data using only its relational capabilities.

  2. Rule 1: Information Rule - All data should be stored in the tables.

  3. Rule 2: Guaranteed access rule - Each unique piece of data should be accessible by table name, primary key, and column name.

  4. Rule 3: Systematic treatment of null values - Null values should be handled consistently.

These rules help guide the development and evaluation of relational database systems.


The next part of the tutorial will dive into popular relational database management systems and how to create a database engine in Python. Let me know if you'd like to proceed.


V. Popular Relational Database Management Systems (RDBMS)

Now that we have understood the concepts behind relational databases, let's look at some popular systems that use these concepts to manage data.


A. Description and Significance of PostgreSQL

PostgreSQL is a powerful, open-source relational database system that is highly extensible and standards-compliant. With PostgreSQL, you have the freedom to define your own data types, build custom functions, or even write code from different programming languages without recompiling your database!

Consider PostgreSQL as an artist's toolkit. You're not only provided with the colors and brushes (read as data types and functions) but also the ability to create your own tools fitting your unique painting style (read as your unique business needs).


B. Description and Significance of MySQL

MySQL is an open-source relational database management system known for its speed and reliability. It is commonly used for web applications and online publishing. Its user-friendly nature makes it a favorite for beginners. MySQL is like a trusted car - it might not have all the latest features, but it's reliable, and it'll get you where you need to go.


C. Description and Significance of SQLite

SQLite is an in-process library that implements a self-contained, serverless, zero-configuration, transactional SQL database engine. Unlike other database systems, SQLite doesn't require a separate server process to operate, making it an excellent database choice for devices with low memory, like smartphones or other IoT devices. Think of SQLite as a notepad. It's lightweight, portable, and gets the job done in environments where deploying heavy solutions isn't practical.


D. Introduction to the SQL Query Language

SQL (Structured Query Language) is the standard language used to communicate with a relational database. SQL is to databases what English is to humans (in most parts of the world). It's a common language that most database systems understand. With SQL, you can do anything from creating a database, creating tables, inserting and updating data, to querying data.


VI. Creating a Database Engine in Python

Python, being an extensible language, allows us to interact with RDBMS using certain packages. Here, we will learn how to create a database engine using SQLite for demonstration.


A. Requirement and Purpose of Creating a Database Engine

In order to interact with a database using Python, we first need to create a database engine. The engine is essentially an interface that allows us to communicate with the database, much like how a web browser is an interface to access the internet.


B. Use of SQLite Database for Illustrative Purposes

We'll use the SQLite database for our example as it is easy to set up, requires no server, and stores the entire database as a single file.


C. Description of the Package SQLAlchemy and its Purpose

SQLAlchemy is a Python library for interacting with relational databases. It provides a full suite of well-known patterns for interacting with databases in a more pythonic and friendly way, without the need to write SQL queries.

from sqlalchemy import create_engine

# Create a database engine to the SQLite database file 'example.db'
engine = create_engine('sqlite:///example.db')

print(engine)

Output: Engine(sqlite:///example.db)

This snippet creates a connection (engine) to a SQLite database file named 'example.db'. If the file doesn't exist, it will be created.


D. Method to Connect to a Database

After creating an engine, we can use it to connect to the database. Connections are created using the connect() method.

connection = engine.connect()
print(connection)

Output: <sqlalchemy.engine.base.Connection object at 0x7f8cd4a22eb8>

The connect() method returns a connection object which we can use to execute queries.


E. Process to Get the Names of Tables within a Database

The engine object's table_names() method is used to retrieve a list of all table names in the database.

tables = engine.table_names()
print(tables)

Output: ['Orders', 'Customers']

In our next part of the tutorial, we will learn how to query these relational databases using Python. Let me know if you're ready to proceed.


VII. Querying Relational Databases in Python

As a data scientist, the real power comes from your ability to manipulate, extract and analyze data. SQL provides an excellent way to select the precise data we need. Now let's explore how we can use Python to send SQL queries and get back results.


A. Introduction to the Concept of SQL Queries

SQL queries are the primary way in which you communicate with a relational database. A SQL query is a question you ask the database. For instance, you could ask, "What is the total amount of all sales made this month?" and the database would return an answer. In SQL, this question could look something like this:

SELECT SUM(SaleAmount) FROM Sales WHERE SaleDate >= '2023-07-01' AND SaleDate < '2023-08-01';

B. Explanation of a Basic SQL Query

A basic SQL query involves the SELECT statement. The SELECT statement is used to select data from a database. The data returned is stored in a result table, called the result-set.


Consider the following example:

SELECT * FROM Customers;

This SQL statement selects all data in the "Customers" table.


C. Introduction to the Workflow of SQL Querying in Python

In Python, using a database connection created with SQLAlchemy, we can send SQL queries to the database using the execute() method on our connection. Here's how it looks:

# Run a SQL query
result_proxy = connection.execute("SELECT * FROM Customers;")

# Fetch all the rows
result_set = result_proxy.fetchall()

In the first line, we're sending a SQL query to the database. The execute() function takes a string of SQL to run and returns a "ResultProxy" that we can use to fetch the results.


D. Step-by-Step Explanation of Executing a Query and Storing Its Results

The execute() method returns a ResultProxy, and we can use the fetchall() method on the ResultProxy to get a list of tuples representing each row. Let's do that:

# Run a SQL query
result_proxy = connection.execute("SELECT * FROM Customers;")

# Fetch all the rows
result_set = result_proxy.fetchall()

# Print each row
for row in result_set:
    print(row)

This will print something like:

('CUST01', 'Acme Corporation', 'John Doe', 'jdoe@acme.com', '(111) 222-3333')
('CUST02', 'Globex Corporation', 'Jane Smith', 'jsmith@globex.com', '(444) 555-6666')


E. Overview of Managing Column Names in the Resulting Data

It's a bit difficult to understand the data without column names. Luckily, we can access them too:

# Get the column names from the ResultProxy
columns = result_proxy.keys()

print(columns)

Output:

['CustomerID', 'CompanyName', 'ContactName', 'Email', 'Phone']


F. Introduction to the Context Manager for Handling Connections

We also need to make sure we close our connections when we're done with them. We can use a Python context manager to ensure our connections are properly cleaned up after use:

# Use a context manager to automatically close our connection after use
with engine.connect() as connection:
    result = connection.execute("SELECT * FROM Customers;")
    rows = result.fetchall()
    columns = result.keys()

print(rows)
print(columns)


VIII. Querying Relational Databases Directly with Pandas

While SQLAlchemy and raw SQL are powerful, there's a simpler way for us to query our data - using pandas!


A. Simplifying the Querying Process Using Pandas

Pandas is a fantastic library for handling and analyzing data. It includes a function, read_sql_query(), that allows us to easily read the results of a SQL query into a DataFrame. Here's how we can use it:

import pandas as pd

# Create a SQL query
query = "SELECT * FROM Customers;"

# Use pandas to execute the SQL query and store the result in a DataFrame
df = pd.read_sql_query(query, engine)

print(df.head())

The read_sql_query() function takes a SQL query and a SQLAlchemy Engine and returns a DataFrame. The DataFrame has all the nice features we're used to with pandas, like a great string representation that includes column names:

  CustomerID           CompanyName   ContactName                 Email             Phone
0      CUST01      Acme Corporation      John Doe        jdoe@acme.com  (111) 222-3333
1      CUST02  Globex Corporation    Jane Smith     jsmith@globex.com  (444) 555-6666

In our next part of the tutorial, we will take a deep dive into the advanced querying methods and how to exploit table relationships for efficient and effective data extraction. Let me know when you're ready to proceed.


IX. Advanced Querying: Exploiting Table Relationships

Exploiting table relationships is a strength of relational databases. If you've structured your database well with appropriate relationships, you can perform very complex queries quite easily. We have touched on table relationships earlier, but now we're going to see how to use them in queries.


A. Reiteration of the Significance of Table Relationships in Relational Databases

As we've mentioned before, in a relational database, tables are interlinked through their keys. This connection allows us to access data spread across different tables as though it was in one place. This feature is incredibly powerful, as it allows us to structure our data in a way that prevents redundancy and maintains data integrity, while still giving us the flexibility to access it as we wish.


Imagine that you're organizing a large party, and you've got all the party data separated into different spreadsheets - one for guests, one for food items, one for drinks, etc. In a non-relational world, if you wanted to know which guest prefers which food, you'd have to manually check two spreadsheets. But if your spreadsheets were relational and connected by a common key (say, guest ID), you could easily retrieve that information as if it was in one place. This is what we can achieve with relational databases.


B. Explanation of Table Linking using 'CustomerID' and 'EmployeeID'

For our synthetic database example, consider that we have a Orders table and a Customers table. The Orders table has a 'CustomerID' column, and each Order row has a 'CustomerID' that corresponds to a Customer in the Customers table.

The 'CustomerID' acts as a bridge between the Orders and Customers table. If we want to know more information about the customer who made a particular order, we don't need to store all the customer information in the Orders table. Instead, we store only the 'CustomerID', and when we need the customer details, we can retrieve them from the Customers table.


C. Introduction to the Concept of JOINing Tables in SQL

SQL provides the JOIN clause that we can use to combine rows from two or more tables, based on a related column between them. There are different types of SQL joins: (INNER) JOIN, LEFT (OUTER) JOIN, RIGHT (OUTER) JOIN, and FULL (OUTER) JOIN. But for this tutorial, let's focus on the most commonly used one, INNER JOIN.


An INNER JOIN picks out rows from both tables that match the condition specified and combines them. If it helps, think of INNER JOIN as the intersection point of two sets, where only the matching elements are included.


D. Use of JOIN to Incorporate Information from Different Tables into a Query


Let's assume we want to get the customer's name for each order. We can use an INNER JOIN to combine data from the Customers and Orders tables:

SELECT Orders.OrderID, Customers.CustomerName
FROM Orders
INNER JOIN Customers ON Orders.CustomerID = Customers.CustomerID;

This SQL statement selects the 'OrderID' from the "Orders" table, and the 'CustomerName' from the "Customers" table, where the 'CustomerID' is matching in both tables.


E. Detailed Walkthrough of an INNER JOIN in Python using Pandas

Let's use our pandas skills to perform the same INNER JOIN operation in Python.

# Create a SQL query
query = """
SELECT Orders.OrderID, Customers.CustomerName
FROM Orders
INNER JOIN Customers ON Orders.CustomerID = Customers.CustomerID;
"""

# Use pandas to execute the SQL query and store the result in a DataFrame
df = pd.read_sql_query(query, engine)

print(df.head())

The output should look like this:

   OrderID   CustomerName
0     1001   John Doe
1     1002   Jane Smith
2     1003   Michael Johnson

As you can see, using the JOIN clause, we've successfully combined data from two tables as if they were in a single table!


That wraps up our tutorial on relational databases, from the basic concepts and examples to querying with Python and pandas. With these tools, you can now interact with relational databases efficiently and make the most out of your data. Happy data wrangling!


Conclusion

Relational databases provide a robust and efficient method for storing, manipulating, and retrieving structured data. They offer numerous advantages including data integrity, flexibility, and the ability to express complex queries. SQL, a language designed for managing data in relational databases, allows for effective communication with databases to perform tasks such as data retrieval, update, insertion, and deletion. Python, with libraries such as SQLAlchemy and pandas, provides powerful tools to interact with relational databases and handle the resulting data in a more pythonic way. With this knowledge, you are well-equipped to handle relational database tasks in your data science journey.

bottom of page