Code icon

The App is Under a Quick Maintenance

We apologize for the inconvenience. Please come back later

Menu iconMenu iconPython & SQL Bible
Python & SQL Bible

Chapter 15: Advanced SQL

15.1 Subqueries

After getting comfortable with SQL basics and diving deep into its querying functionalities in the previous chapters, it's now time to take another step forward into the world of Advanced SQL. This chapter is designed to expose you to the more complex capabilities of SQL, which will help you master the art of managing and manipulating data.

As you move through this chapter, you will uncover the power and flexibility that SQL provides for manipulating and analyzing data on a deeper level. You will learn how to use subqueries to extract data from one or more tables, and how to use advanced joins to combine data from multiple tables based on common columns.

In addition to that, you will be introduced to transactions and their importance in maintaining data consistency and integrity. You will also learn how to create stored procedures, which are reusable code blocks that can be called multiple times with different input parameters.

Furthermore, you will gain insights into how these concepts can be used together to solve real-world problems efficiently. You will learn how to optimize queries for better performance, and how to use indexes to speed up data retrieval.

By the end of this chapter, you will have a much deeper understanding of how SQL works and how it can be used to solve complex data problems. You will be equipped with a powerful set of tools that will enable you to manage and manipulate data efficiently, and you will be ready to take on more challenging SQL tasks with confidence.

Let's get started!

A subquery, also known as an inner query or nested query, is a powerful tool in SQL that allows you to perform more complex queries by using data from another query. Essentially, it is a query within another SQL query, and is used to further restrict the data to be retrieved by returning data that will be used in the main query as a condition.

For example, you could use a subquery to find all customers who have made a purchase in the last month, and then use that data in the main query to retrieve their contact information. This can be particularly useful in situations where you need to perform complex data analysis or retrieve data from multiple tables.

Subqueries can be used with a variety of SQL statements, including SELECT, INSERT, UPDATE, and DELETE, and can be used in combination with a range of operators like =, <, >, >=, <=, IN, BETWEEN etc. With so many possibilities, it's clear that subqueries are an essential tool for any SQL developer looking to take their queries to the next level.

There are two types of Subqueries:

  1. Single Row Subquery: Returns zero or one row.
  2. Multiple Row Subquery: Returns one or more rows.

Let's look at an example:

Suppose you have a database of products with the following structure:

products:
id | product_name | category_id | price

And you want to find out all the products that have a price higher than the average price of all products. You could accomplish this using a subquery like so:

SELECT product_name, price
FROM products
WHERE price > (SELECT AVG(price) FROM products);

In this example, the subquery (SELECT AVG(price) FROM products) calculates the average price of all products. The outer query then uses this average price to return all products that have a price higher than this average.

It's important to note that the subquery is executed first and then the main query is executed. The subquery must always return a value that is used in the main query.

Subqueries can be classified based on their position in the main query.

15.1.1 Scalar Subquery

Scalar subqueries are queries that return a single row with a single column. Scalar subqueries can be used anywhere a single value is expected.

A scalar subquery is a type of query that returns a single row with a single column. Essentially, it is a query within a query, and it can be used anywhere where a single value is required. Scalar subqueries are particularly useful when it comes to analyzing large datasets, as they allow for quick and efficient retrieval of specific information.

For instance, one might use a scalar subquery to determine the average age of a group of people, or to find the maximum value in a particular column. By using scalar subqueries, analysts can gain a deeper understanding of their data and make more informed decisions based on that data.

Example:

SELECT product_name, price
FROM products
WHERE price = (SELECT MIN(price) FROM products);

This query returns the name and price of the product with the minimum price in the table.

15.1.2 Correlated Subquery

A correlated subquery is a type of subquery that uses values from the outer query in its WHERE clause. This means that the subquery is not independent of the outer query and is executed for every row processed by the outer query.

The correlated subquery acts as a filter, helping to extract data that satisfies certain conditions and is useful when you need to retrieve data from two related tables. This type of subquery can also be used to update data in a table based on values from another table. As a result, the correlated subquery can be a powerful tool in database management and is frequently used in complex queries.

Example:

SELECT p1.product_name, p1.price
FROM products p1
WHERE price > (SELECT AVG(p2.price) FROM products p2 WHERE p1.category_id = p2.category_id);

This query returns the products that have a price greater than the average price of products in the same category.

Remember, using subqueries can sometimes lead to inefficient queries. SQL has to run the subquery for each row that might be processed in the outer query, which can lead to long execution times. When writing subqueries, you need to make sure that your query is as efficient as possible. It's often a good idea to try and rewrite your query without a subquery, or even better, to try and write your query so that it only needs to run the subquery once.

Understanding and using subqueries effectively is a key skill in writing advanced SQL queries. The ability to write a query within another query allows you to create complex reports and analytics and to maximize the power of SQL. As always, the best way to learn is through practice, so make sure to experiment with subqueries on your own and see how they can be used in different contexts.

15.1.3 Common Table Expressions (CTEs)

A CTE can be thought of as a temporary table that is defined within the execution scope of a single statement. It's a way of defining subqueries that can be referenced multiple times within another SQL statement.

CTEs are often used when complex or recursive queries need to be performed. For example, if you needed to find all the employees who report to a particular manager, and then find all the employees who report to those employees, a CTE could be used to define a recursive query that would traverse the employee hierarchy.

Another use case for CTEs is when you need to perform multiple subqueries that reference the same data. Instead of writing out the subqueries multiple times, you can define a CTE that encapsulates the subquery logic and then reference it as needed in your main query.

Overall, CTEs provide a way to simplify and modularize complex SQL statements, making them easier to read and maintain over time.

Example:

WITH Sales_CTE (SalesPersonID, NumberOfOrders)
AS
(
  SELECT SalesPersonID, COUNT(OrderID)
  FROM SalesOrderHeader
  GROUP BY SalesPersonID
)
SELECT AVG(NumberOfOrders) as "Average Number of Orders"
FROM Sales_CTE;

This query calculates the average number of orders per salesperson in a company. The CTE is creating a temporary table that counts the number of orders per salesperson. This table is then used to calculate the average number of orders.

CTEs can be particularly useful in complex queries where you need to reference the same subquery multiple times. Instead of writing out the same subquery multiple times, you can define it once in a CTE and then reference that CTE as many times as needed.

This concludes our deep dive into the concept of subqueries. However, it's important to keep in mind that SQL is a vast language, and there's always more to learn. As you become more comfortable with SQL, you'll find that subqueries and CTEs are powerful tools that can help you solve complex problems and create more efficient queries.

15.1 Subqueries

After getting comfortable with SQL basics and diving deep into its querying functionalities in the previous chapters, it's now time to take another step forward into the world of Advanced SQL. This chapter is designed to expose you to the more complex capabilities of SQL, which will help you master the art of managing and manipulating data.

As you move through this chapter, you will uncover the power and flexibility that SQL provides for manipulating and analyzing data on a deeper level. You will learn how to use subqueries to extract data from one or more tables, and how to use advanced joins to combine data from multiple tables based on common columns.

In addition to that, you will be introduced to transactions and their importance in maintaining data consistency and integrity. You will also learn how to create stored procedures, which are reusable code blocks that can be called multiple times with different input parameters.

Furthermore, you will gain insights into how these concepts can be used together to solve real-world problems efficiently. You will learn how to optimize queries for better performance, and how to use indexes to speed up data retrieval.

By the end of this chapter, you will have a much deeper understanding of how SQL works and how it can be used to solve complex data problems. You will be equipped with a powerful set of tools that will enable you to manage and manipulate data efficiently, and you will be ready to take on more challenging SQL tasks with confidence.

Let's get started!

A subquery, also known as an inner query or nested query, is a powerful tool in SQL that allows you to perform more complex queries by using data from another query. Essentially, it is a query within another SQL query, and is used to further restrict the data to be retrieved by returning data that will be used in the main query as a condition.

For example, you could use a subquery to find all customers who have made a purchase in the last month, and then use that data in the main query to retrieve their contact information. This can be particularly useful in situations where you need to perform complex data analysis or retrieve data from multiple tables.

Subqueries can be used with a variety of SQL statements, including SELECT, INSERT, UPDATE, and DELETE, and can be used in combination with a range of operators like =, <, >, >=, <=, IN, BETWEEN etc. With so many possibilities, it's clear that subqueries are an essential tool for any SQL developer looking to take their queries to the next level.

There are two types of Subqueries:

  1. Single Row Subquery: Returns zero or one row.
  2. Multiple Row Subquery: Returns one or more rows.

Let's look at an example:

Suppose you have a database of products with the following structure:

products:
id | product_name | category_id | price

And you want to find out all the products that have a price higher than the average price of all products. You could accomplish this using a subquery like so:

SELECT product_name, price
FROM products
WHERE price > (SELECT AVG(price) FROM products);

In this example, the subquery (SELECT AVG(price) FROM products) calculates the average price of all products. The outer query then uses this average price to return all products that have a price higher than this average.

It's important to note that the subquery is executed first and then the main query is executed. The subquery must always return a value that is used in the main query.

Subqueries can be classified based on their position in the main query.

15.1.1 Scalar Subquery

Scalar subqueries are queries that return a single row with a single column. Scalar subqueries can be used anywhere a single value is expected.

A scalar subquery is a type of query that returns a single row with a single column. Essentially, it is a query within a query, and it can be used anywhere where a single value is required. Scalar subqueries are particularly useful when it comes to analyzing large datasets, as they allow for quick and efficient retrieval of specific information.

For instance, one might use a scalar subquery to determine the average age of a group of people, or to find the maximum value in a particular column. By using scalar subqueries, analysts can gain a deeper understanding of their data and make more informed decisions based on that data.

Example:

SELECT product_name, price
FROM products
WHERE price = (SELECT MIN(price) FROM products);

This query returns the name and price of the product with the minimum price in the table.

15.1.2 Correlated Subquery

A correlated subquery is a type of subquery that uses values from the outer query in its WHERE clause. This means that the subquery is not independent of the outer query and is executed for every row processed by the outer query.

The correlated subquery acts as a filter, helping to extract data that satisfies certain conditions and is useful when you need to retrieve data from two related tables. This type of subquery can also be used to update data in a table based on values from another table. As a result, the correlated subquery can be a powerful tool in database management and is frequently used in complex queries.

Example:

SELECT p1.product_name, p1.price
FROM products p1
WHERE price > (SELECT AVG(p2.price) FROM products p2 WHERE p1.category_id = p2.category_id);

This query returns the products that have a price greater than the average price of products in the same category.

Remember, using subqueries can sometimes lead to inefficient queries. SQL has to run the subquery for each row that might be processed in the outer query, which can lead to long execution times. When writing subqueries, you need to make sure that your query is as efficient as possible. It's often a good idea to try and rewrite your query without a subquery, or even better, to try and write your query so that it only needs to run the subquery once.

Understanding and using subqueries effectively is a key skill in writing advanced SQL queries. The ability to write a query within another query allows you to create complex reports and analytics and to maximize the power of SQL. As always, the best way to learn is through practice, so make sure to experiment with subqueries on your own and see how they can be used in different contexts.

15.1.3 Common Table Expressions (CTEs)

A CTE can be thought of as a temporary table that is defined within the execution scope of a single statement. It's a way of defining subqueries that can be referenced multiple times within another SQL statement.

CTEs are often used when complex or recursive queries need to be performed. For example, if you needed to find all the employees who report to a particular manager, and then find all the employees who report to those employees, a CTE could be used to define a recursive query that would traverse the employee hierarchy.

Another use case for CTEs is when you need to perform multiple subqueries that reference the same data. Instead of writing out the subqueries multiple times, you can define a CTE that encapsulates the subquery logic and then reference it as needed in your main query.

Overall, CTEs provide a way to simplify and modularize complex SQL statements, making them easier to read and maintain over time.

Example:

WITH Sales_CTE (SalesPersonID, NumberOfOrders)
AS
(
  SELECT SalesPersonID, COUNT(OrderID)
  FROM SalesOrderHeader
  GROUP BY SalesPersonID
)
SELECT AVG(NumberOfOrders) as "Average Number of Orders"
FROM Sales_CTE;

This query calculates the average number of orders per salesperson in a company. The CTE is creating a temporary table that counts the number of orders per salesperson. This table is then used to calculate the average number of orders.

CTEs can be particularly useful in complex queries where you need to reference the same subquery multiple times. Instead of writing out the same subquery multiple times, you can define it once in a CTE and then reference that CTE as many times as needed.

This concludes our deep dive into the concept of subqueries. However, it's important to keep in mind that SQL is a vast language, and there's always more to learn. As you become more comfortable with SQL, you'll find that subqueries and CTEs are powerful tools that can help you solve complex problems and create more efficient queries.

15.1 Subqueries

After getting comfortable with SQL basics and diving deep into its querying functionalities in the previous chapters, it's now time to take another step forward into the world of Advanced SQL. This chapter is designed to expose you to the more complex capabilities of SQL, which will help you master the art of managing and manipulating data.

As you move through this chapter, you will uncover the power and flexibility that SQL provides for manipulating and analyzing data on a deeper level. You will learn how to use subqueries to extract data from one or more tables, and how to use advanced joins to combine data from multiple tables based on common columns.

In addition to that, you will be introduced to transactions and their importance in maintaining data consistency and integrity. You will also learn how to create stored procedures, which are reusable code blocks that can be called multiple times with different input parameters.

Furthermore, you will gain insights into how these concepts can be used together to solve real-world problems efficiently. You will learn how to optimize queries for better performance, and how to use indexes to speed up data retrieval.

By the end of this chapter, you will have a much deeper understanding of how SQL works and how it can be used to solve complex data problems. You will be equipped with a powerful set of tools that will enable you to manage and manipulate data efficiently, and you will be ready to take on more challenging SQL tasks with confidence.

Let's get started!

A subquery, also known as an inner query or nested query, is a powerful tool in SQL that allows you to perform more complex queries by using data from another query. Essentially, it is a query within another SQL query, and is used to further restrict the data to be retrieved by returning data that will be used in the main query as a condition.

For example, you could use a subquery to find all customers who have made a purchase in the last month, and then use that data in the main query to retrieve their contact information. This can be particularly useful in situations where you need to perform complex data analysis or retrieve data from multiple tables.

Subqueries can be used with a variety of SQL statements, including SELECT, INSERT, UPDATE, and DELETE, and can be used in combination with a range of operators like =, <, >, >=, <=, IN, BETWEEN etc. With so many possibilities, it's clear that subqueries are an essential tool for any SQL developer looking to take their queries to the next level.

There are two types of Subqueries:

  1. Single Row Subquery: Returns zero or one row.
  2. Multiple Row Subquery: Returns one or more rows.

Let's look at an example:

Suppose you have a database of products with the following structure:

products:
id | product_name | category_id | price

And you want to find out all the products that have a price higher than the average price of all products. You could accomplish this using a subquery like so:

SELECT product_name, price
FROM products
WHERE price > (SELECT AVG(price) FROM products);

In this example, the subquery (SELECT AVG(price) FROM products) calculates the average price of all products. The outer query then uses this average price to return all products that have a price higher than this average.

It's important to note that the subquery is executed first and then the main query is executed. The subquery must always return a value that is used in the main query.

Subqueries can be classified based on their position in the main query.

15.1.1 Scalar Subquery

Scalar subqueries are queries that return a single row with a single column. Scalar subqueries can be used anywhere a single value is expected.

A scalar subquery is a type of query that returns a single row with a single column. Essentially, it is a query within a query, and it can be used anywhere where a single value is required. Scalar subqueries are particularly useful when it comes to analyzing large datasets, as they allow for quick and efficient retrieval of specific information.

For instance, one might use a scalar subquery to determine the average age of a group of people, or to find the maximum value in a particular column. By using scalar subqueries, analysts can gain a deeper understanding of their data and make more informed decisions based on that data.

Example:

SELECT product_name, price
FROM products
WHERE price = (SELECT MIN(price) FROM products);

This query returns the name and price of the product with the minimum price in the table.

15.1.2 Correlated Subquery

A correlated subquery is a type of subquery that uses values from the outer query in its WHERE clause. This means that the subquery is not independent of the outer query and is executed for every row processed by the outer query.

The correlated subquery acts as a filter, helping to extract data that satisfies certain conditions and is useful when you need to retrieve data from two related tables. This type of subquery can also be used to update data in a table based on values from another table. As a result, the correlated subquery can be a powerful tool in database management and is frequently used in complex queries.

Example:

SELECT p1.product_name, p1.price
FROM products p1
WHERE price > (SELECT AVG(p2.price) FROM products p2 WHERE p1.category_id = p2.category_id);

This query returns the products that have a price greater than the average price of products in the same category.

Remember, using subqueries can sometimes lead to inefficient queries. SQL has to run the subquery for each row that might be processed in the outer query, which can lead to long execution times. When writing subqueries, you need to make sure that your query is as efficient as possible. It's often a good idea to try and rewrite your query without a subquery, or even better, to try and write your query so that it only needs to run the subquery once.

Understanding and using subqueries effectively is a key skill in writing advanced SQL queries. The ability to write a query within another query allows you to create complex reports and analytics and to maximize the power of SQL. As always, the best way to learn is through practice, so make sure to experiment with subqueries on your own and see how they can be used in different contexts.

15.1.3 Common Table Expressions (CTEs)

A CTE can be thought of as a temporary table that is defined within the execution scope of a single statement. It's a way of defining subqueries that can be referenced multiple times within another SQL statement.

CTEs are often used when complex or recursive queries need to be performed. For example, if you needed to find all the employees who report to a particular manager, and then find all the employees who report to those employees, a CTE could be used to define a recursive query that would traverse the employee hierarchy.

Another use case for CTEs is when you need to perform multiple subqueries that reference the same data. Instead of writing out the subqueries multiple times, you can define a CTE that encapsulates the subquery logic and then reference it as needed in your main query.

Overall, CTEs provide a way to simplify and modularize complex SQL statements, making them easier to read and maintain over time.

Example:

WITH Sales_CTE (SalesPersonID, NumberOfOrders)
AS
(
  SELECT SalesPersonID, COUNT(OrderID)
  FROM SalesOrderHeader
  GROUP BY SalesPersonID
)
SELECT AVG(NumberOfOrders) as "Average Number of Orders"
FROM Sales_CTE;

This query calculates the average number of orders per salesperson in a company. The CTE is creating a temporary table that counts the number of orders per salesperson. This table is then used to calculate the average number of orders.

CTEs can be particularly useful in complex queries where you need to reference the same subquery multiple times. Instead of writing out the same subquery multiple times, you can define it once in a CTE and then reference that CTE as many times as needed.

This concludes our deep dive into the concept of subqueries. However, it's important to keep in mind that SQL is a vast language, and there's always more to learn. As you become more comfortable with SQL, you'll find that subqueries and CTEs are powerful tools that can help you solve complex problems and create more efficient queries.

15.1 Subqueries

After getting comfortable with SQL basics and diving deep into its querying functionalities in the previous chapters, it's now time to take another step forward into the world of Advanced SQL. This chapter is designed to expose you to the more complex capabilities of SQL, which will help you master the art of managing and manipulating data.

As you move through this chapter, you will uncover the power and flexibility that SQL provides for manipulating and analyzing data on a deeper level. You will learn how to use subqueries to extract data from one or more tables, and how to use advanced joins to combine data from multiple tables based on common columns.

In addition to that, you will be introduced to transactions and their importance in maintaining data consistency and integrity. You will also learn how to create stored procedures, which are reusable code blocks that can be called multiple times with different input parameters.

Furthermore, you will gain insights into how these concepts can be used together to solve real-world problems efficiently. You will learn how to optimize queries for better performance, and how to use indexes to speed up data retrieval.

By the end of this chapter, you will have a much deeper understanding of how SQL works and how it can be used to solve complex data problems. You will be equipped with a powerful set of tools that will enable you to manage and manipulate data efficiently, and you will be ready to take on more challenging SQL tasks with confidence.

Let's get started!

A subquery, also known as an inner query or nested query, is a powerful tool in SQL that allows you to perform more complex queries by using data from another query. Essentially, it is a query within another SQL query, and is used to further restrict the data to be retrieved by returning data that will be used in the main query as a condition.

For example, you could use a subquery to find all customers who have made a purchase in the last month, and then use that data in the main query to retrieve their contact information. This can be particularly useful in situations where you need to perform complex data analysis or retrieve data from multiple tables.

Subqueries can be used with a variety of SQL statements, including SELECT, INSERT, UPDATE, and DELETE, and can be used in combination with a range of operators like =, <, >, >=, <=, IN, BETWEEN etc. With so many possibilities, it's clear that subqueries are an essential tool for any SQL developer looking to take their queries to the next level.

There are two types of Subqueries:

  1. Single Row Subquery: Returns zero or one row.
  2. Multiple Row Subquery: Returns one or more rows.

Let's look at an example:

Suppose you have a database of products with the following structure:

products:
id | product_name | category_id | price

And you want to find out all the products that have a price higher than the average price of all products. You could accomplish this using a subquery like so:

SELECT product_name, price
FROM products
WHERE price > (SELECT AVG(price) FROM products);

In this example, the subquery (SELECT AVG(price) FROM products) calculates the average price of all products. The outer query then uses this average price to return all products that have a price higher than this average.

It's important to note that the subquery is executed first and then the main query is executed. The subquery must always return a value that is used in the main query.

Subqueries can be classified based on their position in the main query.

15.1.1 Scalar Subquery

Scalar subqueries are queries that return a single row with a single column. Scalar subqueries can be used anywhere a single value is expected.

A scalar subquery is a type of query that returns a single row with a single column. Essentially, it is a query within a query, and it can be used anywhere where a single value is required. Scalar subqueries are particularly useful when it comes to analyzing large datasets, as they allow for quick and efficient retrieval of specific information.

For instance, one might use a scalar subquery to determine the average age of a group of people, or to find the maximum value in a particular column. By using scalar subqueries, analysts can gain a deeper understanding of their data and make more informed decisions based on that data.

Example:

SELECT product_name, price
FROM products
WHERE price = (SELECT MIN(price) FROM products);

This query returns the name and price of the product with the minimum price in the table.

15.1.2 Correlated Subquery

A correlated subquery is a type of subquery that uses values from the outer query in its WHERE clause. This means that the subquery is not independent of the outer query and is executed for every row processed by the outer query.

The correlated subquery acts as a filter, helping to extract data that satisfies certain conditions and is useful when you need to retrieve data from two related tables. This type of subquery can also be used to update data in a table based on values from another table. As a result, the correlated subquery can be a powerful tool in database management and is frequently used in complex queries.

Example:

SELECT p1.product_name, p1.price
FROM products p1
WHERE price > (SELECT AVG(p2.price) FROM products p2 WHERE p1.category_id = p2.category_id);

This query returns the products that have a price greater than the average price of products in the same category.

Remember, using subqueries can sometimes lead to inefficient queries. SQL has to run the subquery for each row that might be processed in the outer query, which can lead to long execution times. When writing subqueries, you need to make sure that your query is as efficient as possible. It's often a good idea to try and rewrite your query without a subquery, or even better, to try and write your query so that it only needs to run the subquery once.

Understanding and using subqueries effectively is a key skill in writing advanced SQL queries. The ability to write a query within another query allows you to create complex reports and analytics and to maximize the power of SQL. As always, the best way to learn is through practice, so make sure to experiment with subqueries on your own and see how they can be used in different contexts.

15.1.3 Common Table Expressions (CTEs)

A CTE can be thought of as a temporary table that is defined within the execution scope of a single statement. It's a way of defining subqueries that can be referenced multiple times within another SQL statement.

CTEs are often used when complex or recursive queries need to be performed. For example, if you needed to find all the employees who report to a particular manager, and then find all the employees who report to those employees, a CTE could be used to define a recursive query that would traverse the employee hierarchy.

Another use case for CTEs is when you need to perform multiple subqueries that reference the same data. Instead of writing out the subqueries multiple times, you can define a CTE that encapsulates the subquery logic and then reference it as needed in your main query.

Overall, CTEs provide a way to simplify and modularize complex SQL statements, making them easier to read and maintain over time.

Example:

WITH Sales_CTE (SalesPersonID, NumberOfOrders)
AS
(
  SELECT SalesPersonID, COUNT(OrderID)
  FROM SalesOrderHeader
  GROUP BY SalesPersonID
)
SELECT AVG(NumberOfOrders) as "Average Number of Orders"
FROM Sales_CTE;

This query calculates the average number of orders per salesperson in a company. The CTE is creating a temporary table that counts the number of orders per salesperson. This table is then used to calculate the average number of orders.

CTEs can be particularly useful in complex queries where you need to reference the same subquery multiple times. Instead of writing out the same subquery multiple times, you can define it once in a CTE and then reference that CTE as many times as needed.

This concludes our deep dive into the concept of subqueries. However, it's important to keep in mind that SQL is a vast language, and there's always more to learn. As you become more comfortable with SQL, you'll find that subqueries and CTEs are powerful tools that can help you solve complex problems and create more efficient queries.