DEV Community

Cover image for SQL Practice Problem to Help You Pass Your Interview
Abdisalan
Abdisalan

Posted on • Edited on • Originally published at abdisalan.com

SQL Practice Problem to Help You Pass Your Interview

In the 50+ programming interviews I've done, I've only been asked two SQL questions.

I failed both of those questions.

Sad Pug Dog

While I won't give the questions away, I will give you a problem to practice so that you can succeed where I failed!

This question combines many principles you'll need to quickly solve SQL interview problems.

Hopefully you'll be better prepared than I was 😅

The Dog Database

Imagine you're running a dog shelter and you have a database of dogs and owners. Every dog has one owner, but owners can have many dogs.

Here's the owner and dogs table written in PostgreSQL.

Owner Table

CREATE TABLE owners (
  id SERIAL PRIMARY KEY,
  name VARCHAR(256)
);
Enter fullscreen mode Exit fullscreen mode

Dogs Table

CREATE TABLE dogs (
  id SERIAL PRIMARY KEY,
  owner_id INTEGER REFERENCES owners(id),
  breed VARCHAR(256),
  adopted_on TIMESTAMP
);
Enter fullscreen mode Exit fullscreen mode

Note that there is a one to many relationship between the owners and dogs table. There is one owner id tied to each dog and enforced by a foreign key contraint i.e. you can only use owner ids that actually exist in the owners table.

The Question

Now, that the tables are setup, we can get to the real question.

Write an SQL query that gives the latest dog each owner adopted along with the name of the owner.

Pretty simple right?

Here's some example data to help you out.

owners
 id |  name
----+--------
  1 | PersonA
  2 | PersonB

dogs
 id | owner_id |   breed   | adopted_on
----+----------+-----------+--------------
  1 |        1 | chow chow | 2019-02-03
  2 |        2 | dalmation | 2019-03-07
  3 |        2 | beagle    | 2020-09-21
  4 |        1 | pit bull  | 2020-08-01
Enter fullscreen mode Exit fullscreen mode

The answer to the question should give you a result that looks like this.

 RESULT
name      |   breed  | adopted_on
----------+----------+------------
PersonB   | beagle   | 2020-09-21
PersonA   | pit bull | 2020-08-01
Enter fullscreen mode Exit fullscreen mode

Try this out for yourself first, then I'll go over the answer below. Don't worry about setting this up on your computer! Here's an SQL Fiddle (like CodePen but for SQL) for you test your answer!

http://sqlfiddle.com/#!17/5059f/10

Final Answer

Let's go through this step by step. There's probably a few other ways of doing this but this is mine.

Part 1: Getting the newest dogs

First we find each newest adoption date by each owner.
To do this, I use the max function on the adopted_on column after grouping by owners. I make sure to also get the owner_id, that way we can use it to join on another table.

SELECT owner_id, max(adopted_on) FROM dogs GROUP BY owner_id
Enter fullscreen mode Exit fullscreen mode
RESULT
owner_id |     max
---------+--------------
       1 |  2020-09-21
       2 |  2020-08-01
Enter fullscreen mode Exit fullscreen mode

Part 2: Getting the breed of the newest dogs

Next, we join the last query with the dogs table (itself) to get the breed of the dog and match by the adoption date as well as the owner.

SELECT dogs.breed, dogs.adopted_on FROM dogs
JOIN (
  SELECT owner_id, max(adopted_on) FROM dogs GROUP BY owner_id
) AS newest_dogs
ON
    dogs.owner_id = newest_dogs.owner_id AND
    dogs.adopted_on = newest_dogs.max;
Enter fullscreen mode Exit fullscreen mode
RESULT
  breed    |  adopted_on 
-----------+--------------
  beagle   |  2020-09-21
  pit bull |  2020-08-01
Enter fullscreen mode Exit fullscreen mode

Final: Get the names of the owners

Lastly, we join the result of the last query on the owners table to get their name.

SELECT owners.name, dogs.breed, dogs.adopted_on FROM dogs
JOIN (
  SELECT owner_id, max(adopted_on) FROM dogs GROUP BY owner_id
) AS newest_dogs
ON
  dogs.owner_id = newest_dogs.owner_id AND
  dogs.adopted_on = newest_dogs.max
JOIN
owners ON dogs.owner_id = owners.id;
Enter fullscreen mode Exit fullscreen mode
 FINAL RESULT
name      | breed    | adopted_on
----------+----------+------------
PersonB   | beagle   | 2020-09-21
PersonA   | pit bull | 2020-08-01
Enter fullscreen mode Exit fullscreen mode

Conclusion

Although the question was simple, there were a few tricky queries we had to make! We needed to join tables two times and find the max aggregate on one of the tables.

I hope you learned something from this exercise! If you want to experiment with my final answer, I've also included a SQL Fiddle with the final answer below.

http://sqlfiddle.com/#!17/5059f/9

Thanks for reading! If you want more content, follow me on twitter!

Cover Photo by Berkay Gumustekin on Unsplash

✌️

Top comments (5)

Collapse
 
kamo profile image
KAIDI

Hey,
I think we can do this,

select  name, breed,adopted_on
from
dogs d inner join owners o on 
d.owner_id = o.id
where 
d.adopted_on >=
(
  select max(adopted_on) from dogs
  where d.owner_id = dogs.owner_id
)


`

Collapse
 
abdisalan_js profile image
Abdisalan

Very interesting! That works too!
I never thought to have another query in the where clause for comparison.

d.adopted_on >=
(
  select max(adopted_on) from dogs
  where d.owner_id = dogs.owner_id
)
Collapse
 
kamo profile image
KAIDI

Yeah, you can check the full syntax here
dev.mysql.com/doc/refman/8.0/en/se...

Collapse
 
alexantra profile image
Alex Antra

Having been working with SQL for nearly a decade I got hit with the old 'in what scenario would you use a right join?' last year in an interview. Threw me off! I never use right joins....

Collapse
 
abdisalan_js profile image
Abdisalan

That was exactly what I was feeling when I got my last question! Gotta be prepared for anything these days..