SQL & DDL: A Quick Guide to Creating Tables

SQL & DDL: A Quick Guide to Creating Tables

Relational databases are all about tables. In order to store, organize, or query data; we must first create the tables. In this article, we will use the Data Definition Language (DDL) to learn about the basics of creating tables, choosing data types, adding constraints and default values.

I will be using MySQL for the syntax, but the concepts will apply to any SQL database.

Prerequisites

To get the most out of this article, you should be somewhat comfortable with Database Design and ERDs.

In this article, we're going to create tables using the following ERD for reference:

an ecomerce erd

It's a really really really basic e-commerce database.

  • A user can make many orders and an order belongs to a single user.
  • An order can contain many products and a product can be purchased by many orders.

In a real-world application, this database would be much larger, but this small little database works perfectly for this post. It would probably even work for the world's simplest ecomerce site.

CREATE TABLE

Each box in the ERD will be a table in the database and each row will be a column in the table. Let's start by taking a look at the users.

users

+--------------+
| users        |
+--------------+
| id           |
| email        |
| password     |
| phone_number |
| created      |
+--------------+

We need to CREATE a users table with columns for id, email, password, and phone_number.

CREATE TABLE users (
  id,
  email,
  `password`,
  phone_number,
  created
);

Here we are defining the name of the table users, and name of each column in the table. password is a reserved word in MySQL so we have to surround it with backticks.

It's not enough to just specify the table and column names—we must also specify a data type for each of the columns. What kind of data is an id or an email? For example, if we just wanted to use text to store all of this data, we could make each column TEXT:

CREATE TABLE users (
  id TEXT,
  email TEXT,
  `password` TEXT,
  phone_number TEXT,
  created TEXT
);

This is now valid SQL, but it's not a good practice pretty inefficient to do things this way. The TEXT data type should be used to store large amounts of text.

Ok, so what data type should I use?

What a good question.

INTEGER, VARCHAR, TEXT, BOOLEAN, and DATE are common examples of SQL data types; however, each DBMS will have a different set of data types that you can choose from.

For example, if we wanted to represent a timestamp, an exact date and time:

We might use DATETIME in MySQL, TIMESTAMP in PostgreSQL, and INTEGER in SQLite.

id INTEGER PRIMARY KEY

id is just going to be a unique number to help us identify each individual user. So an INTEGER would be good here.

CREATE TABLE users (
  id INTEGER,
  email,
  `password`,
  phone_number,
  created
);

The id is also the primary key for this table, so we need to specify that after the data type.

CREATE TABLE users (
  id INTEGER PRIMARY KEY,
  email,
  `password`,
  phone_number,
  created
);

VARCHAR

email

email will obviously be a string. In a programming language this would be enough information, but SQL wants to know more. Will it be a constant length string or variable length string? How many characters will the string be?

For an email address, the length might be different for each user. For variable length strings, VARCHAR is usually a good option. With a VARCHAR we should always specify a maximum length, but what's a good maximum length of an email address? 20 characters? 50 characters?

We probably don't want to choose an arbitrary value here, and I can't really think of a good reason to restrict the size of a user's email address. A good choice might be to use 255.

The database will have to allocate some storage for a variable that keeps track of the length of the string. If we limit the size to 255, then the database will only have to allocate 1 byte to keep track of the VARCHAR size. This is the largest value we can use without rolling over to 2 bytes.

Also, when's the last time you saw an email address that was this long?

thisisgoingtobeareallylongstringtogetto255characterswowimonlyat75charactersrightnowthisismoredifficultthanIthoughthowmanynow134okoverhalfwayletstryajokehowdocomputersgetdruk-theytakescreenshots-lol207stillneedsomemorecharactersknockknock-whosthe@gmail.com
CREATE TABLE users (
  id INTEGER PRIMARY KEY,
  email VARCHAR(255),
  `password`,
  phone_number,
  created
);

password

passwords are also going to be variable length strings, do we want to restrict the length of a user's password? Not really, longer passwords are generally more secure.

Wait, we will NEVER store a password in a database. That would be a major security risk. Imagine if the database got hacked and everyone's passwords got leaked. Your reputation would be soiled, you'd have to shut down the application, move back in with your parents and spend the rest of your days streaming League of Legends to twitch at 4am convincing yourself that this could be your new career.

No! We won't let it come to that. Let's just hash the password before be store in the database. We can use bcrypt. Ok, how long's a bcrypt hashed password? I don't know, it's not longer than 255 characters though.

CREATE TABLE users (
  id INTEGER PRIMARY KEY,
  email VARCHAR(255),
  `password` VARCHAR(255),
  phone_number,
  created
);

phone number

Ok, I know this one. It's got to be a number right, I mean the name is literally phone_number. It's tempting to want to use something like INTEGER here, and that would work for a lot of phone numbers. 555 555 5555 could be easily represented as an INTEGER. But what about numbers that start with a leading zero 07555555555. What about those weird international numbers that start with a plus or minus sign. I think it would be safer to store this kind of information as text.

I'm still unsure of the best way of storing a phone number, but the E.164 format is the one suggested by twilio. This standard needs a variable length string with a maximum number of 15 characters.

CREATE TABLE users (
  id INTEGER PRIMARY KEY,
  email VARCHAR(255),
  `password` VARCHAR(255),
  phone_number VARCHAR(15),
  created
);

TIMESTAMP

created

created will represent the exact moment in time that the account was created. In MySQL both DATETIME AND TIMESTAMP use the following format ‘YYYY-MM-DD hh:mm:ss'.

MySQL converts TIMESTAMP values from the current time zone to UTC for storage, and back from UTC to the current time zone for retrieval. This is nice because now the application doesn't have to manage any time conversions.

Try to avoid handling time conversions ever, they're difficult and programmers can't be trusted to do things correctly. Just let the database do it.

CREATE TABLE users (
  id INTEGER PRIMARY KEY,
  email VARCHAR(255),
  `password` VARCHAR(255),
  phone_number VARCHAR(15),
  created TIMESTAMP
);

Ok, that's the users table done. Now let's create the next table, the orders table.

orders

+--------------+
| orders       |
+--------------+
| id           |
| user_id      |
| created      |
+--------------+

In this example, the orders table only has three columns, an id, a user_id which is a foreign key, and a created timestamp. So we already know enough to get this far:

CREATE TABLE orders (
  id INTEGER PRIMARY KEY,
  user_id,
  created TIMESTAMP
);

FOREIGN KEY

user_id

The user_id is a foreign key. It allows us to represent that one-to-many relationship needed here. But how do we represent a foreign key?

1. Make the foreign key the exact same type as the primary key it's associated with. users.id is an INTEGER so orders.user_id also needs to be an INTEGER.

CREATE TABLE orders (
  id INTEGER PRIMARY KEY,
  user_id INTEGER,
  created TIMESTAMP
);

2. Make the foreign key constraint explicit using the following syntax:

FOREIGN KEY (foreign_key) REFERENCES other_table(primary_key)

Which would look like this:

CREATE TABLE orders (
  id INTEGER PRIMARY KEY,
  user_id INTEGER,
  created TIMESTAMP,
  FOREIGN KEY (user_id) REFERENCES users(id)
);

3. Define the Referential Action.

Referential Action

Imagine a situation where you have a user in the database that has made 3 orders.

+----+-------------+
| id | email       |
+----+-------------+
|  1 | me@me.me    |
|  2 | sam@sam.sam |
+----+-------------+

+----+---------+---------------------+
| id | user_id | created             |
+----+---------+---------------------+
|  1 |       2 | 2020-02-02 15:15:58 |
|  2 |       2 | 2020-02-02 15:16:05 |
|  3 |       2 | 2020-02-02 15:16:06 |
+----+---------+---------------------+

That user decides to delete their account, so we delete that user's row out of the users table. What do we do with all of the orders related to that user? We can't just leave them there, they need to reference a user_id and we've just deleted that user.

A Referential Action tells the DBMS what to do when an update or delete operation effects the parent-child relationship columns. So what happens if the primary key gets updated or deleted.

The three most common actions that I've used are CASCADE, SET NULL, and RESTRICT. In the example above, here's what they would do:

CASCADE

CASCADE would delete all of the user's orders so their complete history would be wiped clean.

+----+-------------+
| id | email       |
+----+-------------+
|  1 | me@me.me    |
+----+-------------+

+----+---------+---------------------+
| id | user_id | created             |
+----+---------+---------------------+
+----+---------+---------------------+

SET NULL

SET NULL would set all of the orders' user_ids to null. So the orders would still exist, but not belong to any user.

+----+-------------+
| id | email       |
+----+-------------+
|  1 | me@me.me    |
+----+-------------+

+----+---------+---------------------+
| id | user_id | created             |
+----+---------+---------------------+
|  1 |   NULL  | 2020-02-02 15:15:58 |
|  2 |   NULL  | 2020-02-02 15:16:05 |
|  3 |   NULL  | 2020-02-02 15:16:06 |
+----+---------+---------------------+

RESTRICT

RESTRICT would not allow a user to be removed from the database as long as they had at least one order in the orders table.

+----+-------------+
| id | email       |
+----+-------------+
|  1 | me@me.me    |
|  2 | sam@sam.sam |
+----+-------------+

+----+---------+---------------------+
| id | user_id | created             |
+----+---------+---------------------+
|  1 |       2 | 2020-02-02 15:15:58 |
|  2 |       2 | 2020-02-02 15:16:05 |
|  3 |       2 | 2020-02-02 15:16:06 |
+----+---------+---------------------+

If this action were in place, we would have to delete all of the user's orders before being able to delete the user.

For this site, let's go with CASCADE and just delete everything, all of the orders. Actually, on second thought that seems too risky even for me. We'll add RESTRICT and not allow anyone to delete their data. We could always decide to add a deleted column to “fake” delete an account.

CREATE TABLE orders (
  id INTEGER PRIMARY KEY,
  user_id INTEGER,
  created TIMESTAMP,
  FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE RESTRICT
);

order_products

On to the next table order_products.

+--------------+
|order_products|
+--------------+
| id           |
| order_id     |
| product_id   |
+--------------+

This is the join table between the products and order that allow for the many-to-many relationship. Wait, this table references a product_id, but we haven't created that table yet. We can't setup the foreign key constraints correctly until the products table has been created.

We also have to think about these things when inserting data into the tables. The products and orders must exist before an order_products row can be created.

products

+--------------+
|  products    |
+--------------+
| id           |
| name         |
| price        |
| description  |
| image        |
+--------------+

`Reserved Words`

name is a reserved word in MySQL, so we either have to use a different name for our column, or use backticks around the name.

CREATE TABLE products (
  id INTEGER PRIMARY KEY,
  `name` VARCHAR(255),
  price,
  description,
  image
);

Money

price

price is going to have to store the price of any product, like 10.95. Looks like a float to me, so should we just use the FLOAT type?

price FLOAT

ABSOLUTELY NOT! NEVER STORE MONEY AS A FLOAT!

floats do not represent decimal numbers precisely. For example, javascript uses IEEE 754 Standard for Floating-Point Arithmetic. If you try adding 0.1 + 0.2 in a language like this, you'll end up with 0.30000000000000004. Go ahead and try right now. Open up dev tools on this site and enter 0.1 + 0.2 in the console. 0.30000000000000004!

That kind of precision is fine for some arithmetic, but not for money. Never use a FLOAT any value where complete precision and accuracy is important.

What do we use then? We could use a VARCHAR and just sore the decimal number as a string, that would solve the precision issue “10.95”. Or we could multiply the amount by 100 and store 1095 as an INTEGER since we don't have a decimal number any more. This is basically storing the value as cents instead of dollars.

Using INTEGER would also allow us to perform math equation pretty easily like finding the average price of all products.

There is another type we can use though, the DECIMAL type. This stores decimal numbers with exact precision, we just have to specify precision and the scale values. DECIMAL(precision, scale)

So DECIMAL(8,2) is able to store any value with five digits and two decimals, his will range from -999,999.99 to 999,999.99.

CREATE TABLE products (
  id INTEGER PRIMARY KEY,
  `name` VARCHAR(255),
  price DECIMAL(8,2),
  description,
  image
);

TEXT

description

The product has a name that can be a variable length piece of text up to 255 characters. description is also a variable length string, but it's probably going to be much longer than anything else we've seen so far.

MySQL has a few types that are designed to work with larger amounts of text data TEXT (max 65,535 bytes), MEDIUMTEXT (max 16,777,215 bytes), and LONGTEXT (4,294,967,295 bytes). VARCHAR‘s limit is usually 65,535 which is the same as TEXT.

65,535 bytes is probably long enough for a description for most products on a site. 255 bytes might even be enough. We're going to use TEXT here because it has enough space and is well suited for this kind of text data. Unlike VARCHAR, the contents of TEXT are stored separately from the rest of the row.

CREATE TABLE products (
  id INTEGER PRIMARY KEY,
  `name` VARCHAR(255),
  price DECIMAL(8,2),
  description TEXT,
  image
);

Files

image

Most SQL DBMS have a BLOB type that stands for Binary Large Object. Using this type would allow us to store binary object representations for things like images or videos.

CREATE TABLE products (
  id INTEGER PRIMARY KEY,
  `name` VARCHAR(255),
  price DECIMAL(8,2),
  description TEXT,
  image BLOB
);

However, often we're using a MySQL database to store data for a web app. These applications have a different plan for image storage. A small application might store the database on the same machine as the server and application code. With this type of setup, we might store the image somewhere in the server's file system and store the file path in the database. So a VARCHAR would work well here.

In a larger application, we might store the image on a completely different server and store the URL in the database. For example, we could process the image into different sizes for different devices, distribute those images over a CDN, and our database wouldn't know anything other than a url and the different sizes that were available.

In this example, we'll assume that we're storing the single image on another server, and we'll just store the url.

CREATE TABLE products (
  id INTEGER PRIMARY KEY,
  `name` VARCHAR(255),
  price DECIMAL(8,2),
  description TEXT,
  image_url VARCHAR(255)
);

order_products

And just for completeness, here's the order_products.

+--------------+
|order_products|
+--------------+
| id           |
| order_id     |
| product_id   |
+--------------+
CREATE TABLE order_products (
  id INTEGER PRIMARY KEY,
  order_id INTEGER,
  product_id INTEGER,
  FOREIGN KEY (order_id) REFERENCES orders(id) ON DELETE RESTRICT,
  FOREIGN KEY (product_id) REFERENCES products(id) ON DELETE RESTRICT
);

No surprises there.

Default Values and AUTO_INCREMENT

Creating a new order for a user table might look something like this:

INSERT INTO orders (id, user_id, created)
VALUES (1, 1, NOW());

Here we are creating an order with the id 1, that belongs to user 1 and was created NOW.

Note, NOW() is a function that you can call in MySQL to get the current timestamp in YYYY-MM-DD hh:mm:ss format. You can test this about by running SELECT NOW(); in your database.

Creating another order for a different user might look like this:

INSERT INTO orders (id, user_id, created)
VALUES (2, 10, NOW());

If we keep creating orders like this, we will see a pattern emerge. created will always be NOW() and id will always be incrementing. We need to manually tell the database which user_id to use, but the id and created values could be managed by the database itself.

This isn't just us being lazy, although it is a little bit like that. It's safer to have the database manage these things for us so we don't accidentally mess things up.

Let's change the CREATE TABLE statement for orders:

CREATE TABLE orders (
  id INTEGER PRIMARY KEY AUTO_INCREMENT,
  user_id INTEGER,
  created TIMESTAMP DEFAULT NOW(),
  FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE RESTRICT
);

The AUTO_INCREMENT attribute has been added to id. If we don't manually input a value for id, MySQL will manage the value for us. Pretty much every primary key should have the AUTO_INCREMENT value.

created now has a DEFAULT clause. This can be added to any column to specify a default value for that column. This is the value that will be used if we don't provide one during an insertion. The default value in this case will be the return value of NOW(), which will be the exact time that the order was created.

With these two things added to our table, we really only to provide the user_id when inserting a value:

INSERT INTO orders (user_id)
VALUES (1);

NOT NULL

Our tables are really starting to look complete, but there's a big issue that we haven't addressed yet. What if we accidentally forget to provide a value for he user_id? What if we add an order and only provide a created timestamp?

INSERT INTO orders (created)
VALUES (NOW());

That's ok, the id will be auto incremented, but what will the user_id be?

+----+---------+---------------------+
| id | user_id | created             |
+----+---------+---------------------+
|  1 |    NULL | 2020-02-02 13:09:10 |
+----+---------+---------------------+

We could also achieve this result by passing in NULL as the value:

INSERT INTO orders (user_id)
VALUES (NULL);

NULL means “a missing unknown value”
https://dev.mysql.com/doc/refman/8.0/en/working-with-null.html

NULL can be handy when we need to represent the absence of a value, but it's not really helpful in this situation. How can an order exist without a user? Who created the order? Who's paying for it and where will it get shipped to?

Ok, don't panic, we can fix this. When we don't want to allow the absence of a value like this, we can add a NOT NULL constraint to that column.

CREATE TABLE orders (
  id INTEGER PRIMARY KEY AUTO_INCREMENT NOT NULL,
  user_id INTEGER NOT NULL,
  created TIMESTAMP DEFAULT NOW() NOT NULL,
  FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE RESTRICT
);

Now the id, user_id and created column are always guaranteed to have a value. The values can never be set to NULL. And in this case, it's exactly what we want. There is no situation where having any of these columns as NULL would make sense.

The primary key can never be NULL, it's never allowed. So MySQL implicitly adds a NOT NULL constraint to primary key columns. So the previous example is identical to the following one as far as MySQL is concenerned.

CREATE TABLE orders (
  id INTEGER PRIMARY KEY AUTO_INCREMENT,
  user_id INTEGER NOT NULL,
  created TIMESTAMP DEFAULT NOW() NOT NULL,
  FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE RESTRICT
);

There are few columns in this database that would be ok as NULL. Like the user's phone number, maybe that could be an optional value, but username, password, and created will all need to be NOT NULL

All The Tables

Here is the final state of all the tables:

CREATE TABLE users (
  id INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT,
  email VARCHAR(255) NOT NULL,
  `password` VARCHAR(255) NOT NULL,
  phone_number VARCHAR(15),
  created TIMESTAMP NOT NULL DEFAULT NOW()
);

CREATE TABLE orders (
  id INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT,
  user_id INTEGER NOT NULL,
  created TIMESTAMP NOT NULL DEFAULT NOW(),
  FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE RESTRICT
);

CREATE TABLE products (
  id INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT,
  `name` VARCHAR(255) NOT NULL,
  price DECIMAL(8,2) NOT NULL DEFAULT 0,
  description TEXT,
  image_url VARCHAR(255)
);

CREATE TABLE order_products (
  id INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT,
  order_id INTEGER NOT NULL,
  product_id INTEGER NOT NULL,
  FOREIGN KEY (order_id) REFERENCES orders(id) ON DELETE RESTRICT,
  FOREIGN KEY (product_id) REFERENCES products(id) ON DELETE RESTRICT
);

DROP TABLE

There may come a time when you want to remove one of the tables you created. This doesn't usually happen in production, but will definitely happen in development. To drop a table just write:

DROP TABLE tablename;

That's it. Just make sure you drop the tables in an order that makes sense for the depenedencies. For example, we could drop all of the tables we just created by runnin the drop statemtents in the reverse order that they were created:

DROP TABLE IF EXISTS order_products;
DROP TABLE IF EXISTS products;
DROP TABLE IF EXISTS orders;
DROP TABLE IF EXISTS users;

order_products depends on products and orders. orders depends on users. Deleting the tables this way ensures that we won't get dependency error messages.

ALTER TABLE

We've created the users and just realized that we didn't make a column to add the user's name. How are we supposed to spam our users effeciently if we don't know their names? We could just drop the table and recreate it. With a name column, but that would also delete all of the data. Instead, let's alter the table.

ALTER TABLE users
ADD COLUMN full_name VARCHAR(255) NOT NULL DEFAULT 'Mary Poppins';

We start the statement with ALTER TABLE tablename and then follow that with the changes we want to make. The alter table statement can do a lot of things and a full list for MySQL can be found here. In this case, we're just adding a new column.

Summary

We covered a lot in this article.

  • Creating, dropping, and altering tables.
  • Data types
  • Foreign key constraints
  • Default values
  • Not null constraints

Hopefully, you now have enough information to get started creating tables for your database.

Find an issue with this page? Fix it on GitHub