Optimizing Laravel Eloquent queries

Published: Oct 22, 2023 by C.S. Rhymes

Here are a few tips and tricks that I have learnt along the way to help optimize your database queries in Laravel. Sometimes, little changes can add up and make a big difference overall.

Select only the fields you need

Adding a select() to your query means that you only retrieve the fields that you are going to use, rather than returning every field in your model. If your view doesn’t need the field then save some memory by only fetching and returning the fields you need.

Rather than:

$users = User::get();

You can select only the fields you want, such as id, name and email:

$users = User::query()
    ->select('id', 'name', 'email')
    ->get();

Specifying a select statement

Pluck the field you need

If you only need one field from the models then you can use pluck() to just return that one field. This returns an array instead of a collection of models.

$userNames = User::query()->pluck('name');

If you want a specific key for your array, such as the user’s id, then you can set that field name as the second parameter.

$userNames = User::query()->pluck('name', 'id');

Retrieving a list of column values

Selecting specific fields in relations

When fetching relations you can specify which fields you want to return in the relation. Here we specify that we only want the role id and name returned for the related roles models.

User::query()->with('roles:id,name')->get();

Eager loading specific columns

Let’s say you have an audit table that records each time a user logs in. To get the latest audit you may try and use an existing relationship, such as User has many audits, then filtering them to get the latest item. Instead of doing this, Laravel has a Has One Of Many relationship that you can use to always return the latest or oldest.

public function latestAudit()
{
    return $this->hasOne(Audit::class)->latestOfMany();
}

Then you can use it as follows:

$user = User::query()
    ->with('latestAudit')
    ->first();
$user->latestAudit; // A single Audit model.

Has one of many

Creating indexes

Consider adding indexes to columns that are frequently searched. One example could be searching for existing users in our Laravel app by their name so we can tag them in a comment.

Here we can create an index on the name column in our users table when it is defined in the migration:

$table->string('name')->index();

If we are adding an index to an existing column then you can do it as follows:

$table->index('name');

Another similar example is if you have two fields that are frequently used together. Instead of name, we could have first_name and last_name in our users table. To add a compound index we can do the following in our migration:

$table->index(['first_name', 'last_name']);

Then when we use a where with both fields it should use this compound index:

$users = User::query()
    ->where('first_name', 'like', "{$search}%")
    ->orWhere('last_name', 'like', "{$search}%")
    ->get();

Creating indexes

Where like searches and indexes

It’s tempting to write where statements that search for any rows that contain a string. For example, get any users where their name contains a provided search term, using the % as a wildcard in the search:

$users = User::query()
    ->where('name', 'like', "%{$search}%")
    ->get();

This means that although the index for name exists, the database can’t use it.

If you only need to search strings beginning with the search term, instead of containing the search term, then the database should be able to use your index and return the results faster.

$users = User::query()
    ->where('name', 'like', "{$search}%")
    ->get();

Full text index

If you have a large field of text, something like a biography field on your user, then you could try adding a full text index to the field in a database migration. Just a note though, this is not supported in SQLite.

$table->fullText('biography');

Once you have the full text index you can then use whereFullText() instead of a where like query.

$users = User::query()
    ->whereFullText('biography', 'developer')
    ->get();

Full text where clauses

Pagination

One thing to consider with most of the previous examples is that we are getting all of the results by using get().

This may be fine when there are a few rows in your database, but when you have hundreds or thousands of rows then it will become slow trying to fetch all the rows from the table.

Normally you will only want to return a set of results at a time. You can do this using skip() and take() to limit the results like follows:

$users = User::query()
    ->skip(10)
    ->take(10)
    ->get();

But this can become cumbersome quite quickly, having to keep track of how many to skip.

Instead, you can use paginate() to return a set of results. Laravel automatically detects the page number from the query string so it knows what set of results to display at a time. We can pass in an integer into the pagintate() method to define how many to return, for example this will return 10 results per page.

$users = User::query()
    ->paginate(10);

Database Pagination

Photo by JESHOOTS.com on StockSnap

Laravel Eloquent Database

Share

Latest Posts

Using when with the Laravel Http Client
Using when with the Laravel Http Client

Here’s a little tip I discovered that I haven’t seen documented anywhere. You can use when() and unless() with the Laravel Http client.

New book announcement!
New book announcement!

Announcing the new book, The Little-Astwick Mysteries - Trouble at the church, by C.S. Rhymes. It is now available for pre-order on the Amazon Kindle store for £2.99, with the release date of the 1st February 2024.

Using prettier to format your Jekyll theme
Using prettier to format your Jekyll theme

I have been using prettier for a few years to automatically format code, especially JavaScript and TypeScript projects, as it helps standardise the output on a shared code project. I have maintained a few different Jekyll themes over the years and wanted to use the power of prettier to automatically format code consistently.

How NOT to make a website

How NOT to make a Website

By C.S. Rhymes

From £2.49

Nigel's Intranet Adventure

Nigel's Intranet Adventure

By C.S. Rhymes

From £2.99