Skip to main content
Build the future with Agentforce at TDX in San Francisco or on Salesforce+ on March 5–6. Register now.

Move from SQL to SOQL

Learning Objectives

After completing this unit, you’ll be able to:

  • Understand the benefits of the Force.com objects.
  • Identify what is similar and different between SQL and SOQL.
  • Build a simple SOQL statement using Workbench.
  • Build more complicated relationship queries.
  • Build aggregated queries.

Understanding Force.com Objects

The Force.com platform provides a powerful database, with many features that make it quick and easy to create applications. Having dealt with SQL Server, you know that data is stored in tables and rows. The database in Force.com, on the other hand, uses objects to store data. Objects contain all the functionality you expect in a table, with additional enhancements that make them more powerful and versatile. Each object comprises a number of fields, which correspond to columns in a database. Data is stored in records of the object, which correspond to rows in a database. But wait... there's more!

There are two types of objects:

  1. Standard Objects — These are objects that come baked-in with Salesforce. Some common CRM objects include Accounts, Contacts, Opportunities and Leads.
  2. Custom Objects — These are new objects you create to store information unique to your application. Custom objects extend the functionality that standard objects provide. For example, if you’re building an app to track product inventory, you might create custom objects called Merchandise, Orders or Invoices.

As you probably guessed, objects can have relationship fields that define how records in one object relate to records in another object. These are essentially primary and foreign keys, but they are much more flexible, making it easier to design and implement your data model.

Whether they’re standard or custom, Force.com objects not only provide structure for storing data but also power the interface elements that allow users to interact with the data, such as tabs, the layout of fields on a page, and lists of related records. For standard functionality there's no need to implement an ORM, write a UI for CRUDing data or build tables. This standard functionality is provided by the platform automatically for you. Objects also have built-in support for features such as access management, validation, formulas, and history tracking. All attributes of an object are described with metadata, making it easy to create and modify records either through a visual interface or programmatically.

As you can see, objects are a lot more than simply containers for storing data. They provide a rich set of functionality that frees you up to focus on building the features unique to your application. For more information on how to create custom objects, fields, relationships and much more, check out the Data Modeling module.

Similar but Not the Same

As a .NET developer, chances are pretty high that you’re comfortable working with SQL Server. And chances are also high that you’re familiar with writing ad hoc queries using SQL. So we thought the best way to introduce you to a similar language designed specifically for Salesforce called SOQL, or Salesforce Object Query Language, was to compare the two.

The first thing to know is that although they’re both called query languages, SOQL is used only to perform queries with the SELECT statement. SOQL has no equivalent INSERT, UPDATE, and DELETE statements. In the Salesforce world, data manipulation is handled using a set of methods known as DML (Data Manipulation Language). We’ll talk more about DML shortly. For now, you just need to know how to query Salesforce data using the SELECT statement that SOQL provides.

One big difference you’ll notice right away is that SOQL has no such thing as SELECT *. Because SOQL returns Salesforce data, and that data lives in a multi-tenanted environment where everyone is kind of "sharing the database,” a wildcard character like * is asking for trouble. Honestly speaking, it’s just too easy to fire up a new SQL query and type in SELECT * FROM SOME-TABLE, especially when you don’t know what the table field names are. This action could severely impact other tenants in your shared environments. Would you mow your lawn Sunday morning at 7? That’s just rude and disrespectful.

In SOQL, you specify each field name to return. Beyond that, the SELECT statement that SOQL provides is similar to SQL. You’ll find writing SOQL queries pretty easy. But here’s what you need to know: Although SQL and SOQL are similar, they’re not the same. SOQL doesn’t support some of the more advanced features that the SQL SELECT statement does. But on the Salesforce platform, you really don’t need all those extra features. SOQL offers just what you need in a way that makes you feel right at home.

Building a Query with Workbench

Speaking of writing SOQL queries, you might now be thinking, "Well, how do I do that?" One easy way to get started is to use the Salesforce web-based tool called Workbench. We know how much .NET developers love tools, and this powerful tool offers many ways for administrators and developers to access their orgs using Force.com APIs.

For now, we’ll focus on using Workbench to build SOQL queries, but if you have some free time, browse around Workbench and check out all that it offers. We think you’ll like it. Sign up for a free Developer Edition (DE) org and then:

  1. Navigate to https://workbench.developerforce.com/login.php.
  2. For Environment, select Production.
  3. Select the latest API version from the API Version drop-down menu.
  4. Accept the terms of service, and click Login with Salesforce.
  5. Enter your login credentials and then click Log in.
  6. To allow Workbench to access your information, click Allow.
  7. After logging in, select queries > SOQL Query.
  8. Select Account as the object.

Notice that you chose an object and not a table. Data is persisted inside of objects. Actually, they’re called sObjects, as in Salesforce Objects, and they’re tightly integrated with the Force.com platform, which makes them easy to work with.

  1. Hold down the Ctrl key and select CreatedDate, Name, Phone, Type from the list of Fields. As you select the object and fields, the SOQL query is created for you in the text box. It looks like the following:
    SELECT CreatedDate, Name, Phone, Type FROM Account
    
  2. Click Query to see the results returned as a list. In the results, look at the value returned in the CreatedDate field.

The bad news here is that datetime fields in Salesforce are just as complicated and hard to work with in SOQL as in SQL. The good news is that Salesforce offers several date functions that make working with them in SOQL less painful. And while you're poking around the documentation, check out how SOQL handles currency fields, because they too are a little different, especially for orgs that deal with multiple currencies.

Filtering Your Results

SOQL has only two required clauses: SELECT and FROM. The WHERE clause is optional. But as a good developer (which we’re sure you are), you want to include a WHERE clause for almost every query you write. No sense returning more data than needed.

Once again, the easiest way to see how this works is to use Workbench.

  1. After logging in to Workbench, select queries > SOQL Query.
  2. Select Contact as the object.
  3. Hold down the Ctrl key and select AccountId, Email, Id, LastName from the list of fields. As you select the object and fields, the SOQL query is created for you in the text box.
  4. Click Query to see the results returned as a list. It looks like the following:
    SELECT AccountId, Email, Id, LastName FROM Contact
    

This query returns all contacts in your org. For a development org, this list might be small. But for most real-world orgs, the number of contacts returned could be thousands, which is why you should always consider filtering your SOQL queries with a WHERE clause, especially those used in Apex code.

  1. From the Filter results by drop-down list, select Email.
  2. Tab to the next field and click the arrow in the drop-down box for a list of possible operators.
  3. Select contains from the list, and then tab to the last field and enter .net. The constructed query looks like the following:
    SELECT AccountId, Email, Id, LastName FROM Contact WHERE Email LIKE '%.net%'
    

Did you notice what happened? Your constructed query doesn’t include the word contains. Instead, it uses LIKE. The query also includes the required single quotes (because Email is a text field), as well as leading and trailing percentage signs to indicate that it’s a wildcard search.

Note: Using wildcards, especially leading and trailing wildcards such as in the above example, isn’t considered good practice. You learn more about how to build efficient queries in a later unit, but for now keep in mind to avoid wildcard searches such as these whenever possible.

  1. While you are at it, go ahead and order the results by LastName by selecting LastName from the Sort results by drop-down list.
  2. Leave the other defaults of A to Z and Nulls First. Your query now looks like this:
    SELECT AccountId,Email,Id,LastName FROM Contact
      WHERE Email LIKE '%.net%' ORDER BY LastName ASC NULLS FIRST
    
  3. Click Query to return the results as an ordered list.

The big thing we want you to notice in the results are the two ID fields: AccountId and Id. These fields contain a unique 18-character string that were assigned by the platform when the Account and Contact records were created. The AccountId field is associated with the Account record that this particular Contact was assigned to. In SQL terms, it is a foreign key relationship. The Id field is related to the contact. In SQL terms, it represents the primary key.

While we’re bringing up foreign key relationships, you might now be wondering how you join tables in SOQL. The short answer is you don’t. SOQL has no equivalent JOIN clause. But don’t worry, because we think that’s a good thing.

A Different Kind of Join

We're sure you won’t be surprised to hear this, but Salesforce does things a little differently when it comes to combining objects, or tables, as you’re used to thinking of them. Rather than combine tables with a JOIN clause, you write what is known as relationship queries.

"And what are those?", we can almost hear you saying. Glad you asked.

Salesforce uses a parent-child relationship to combine two objects. Just like in SQL, SOQL uses a foreign key to relate these two objects, but in SOQL, the query syntax is different. We’re not going to lie. At first, using this new syntax probably feels a little weird because you’re working with objects instead of rows. But once you get used to the basics, you’ll find writing relationship queries much easier than the joins you write in SQL.

SOQL has two basic relationship query types that you need to remember:

  • Child-to-parent
  • Parent-to-children

Each one works differently. Because you’ve already seen some fields for the Account and Contact objects, which are most commonly combined, we’ll start with them. The important thing to know for now is that Account is the parent, and Contact is the child.

Write a Child-to-Parent Query

Let’s say that you wanted to write a query that returns Account and Contact information. Your first option is to write a child-to-parent query. This relationship query uses "dot notation" to access data from the parent, that is, a period separates the relationship name from the name of the field being queried.

To see how this works, we’ll walk through writing a relationship query that returns a list of contacts, including the name of the associated account. But this time, instead of using Workbench, we’ll use the Query Editor tab in Developer Console.

  1. Click SetupSetup, then click Developer Console.
  2. In Developer Console, click the Query Editor tab in the bottom pane.
  3. Delete the existing code, and insert the following snippet:
    SELECT FirstName, LastName, Account.Name FROM Contact
    
  4. Click Execute.
  5. The query results include three columns. Scroll through the results. Some results under the Account.name field are null because not all contacts are related to an account.

Because we know you’re probably still thinking in SQL terms, imagine this scenario. If you had two SQL tables named Account and Contact with a one-to-many relationship between the tables, how would you go about writing a SQL query that returned this same information?

You would use a join, of course. But in this case, you need a right outer join, because you want an equivalent query that returns all contacts, even those not related to an account. And that might look something like the following:

    SELECT c.FirstName, c.LastName, a.Name FROM Account a
    RIGHT JOIN Contact c ON (c.AccountId = a.Id)

So now we want you to go back and look at the equivalent SOQL query, which was, in case you don’t remember:

    SELECT FirstName, LastName, Account.Name FROM Contact

The SOQL query looks much simpler, don’t you think?

To be honest, relationship queries can get a little tricky sometimes, especially when figuring out what the relationship name is for custom objects. To learn more about converting SQL queries into SOQL queries, check out this hands-on training video.

One super important thing to be aware of with these types of queries is that you can traverse five levels up with the dot notation. So you can go from the child to the parent to the grandparent to the great grandparent, and so on.

Writing Parent-to-Children Queries

Parent-to-children queries also use the relationship name, but they use it in what is called a nested select query. As with the last query type, it’s better to explain this by showing you an example.

This time we’ll write a query from the parent Account object and have it include a nested query that also returns information about each associated contact.

  1. In Developer Console, click the Query Editor tab in the bottom pane.
  2. Delete the existing code, and insert the following snippet:
    SELECT Name, (Select FirstName, LastName FROM Contacts) FROM Account
    

The relationship name inside the nested query uses the plural name Contacts, as opposed to Contact. This detail is important to understand, and it’s the part that usually trips people up. When working with relationship queries, the parent-to-child relationship name must be a plural name.

When working with custom objects, the relationship name is not only plural, but it is appended with two underscores and an r. For example, the relationship name for the custom object My_Object__c is My_Objects__r.

  1. Click Execute.
  2. The query results include two columns. Notice in the following image how the results under the Contacts column are displayed. Because each account is typically associated with multiple contacts, the first and last names are displayed in JSON format.

SOQL Query Results displayed in Developer Console

Once again, let’s look at how that same query is written in SQL. For this query, the SQL equivalent is a left outer join similar to the following:

    SELECT a.Name, c.FirstName, c.LastName
    FROM Account a
    LEFT JOIN Contact c ON (a.Id = c.AccountId)

The SQL query gets all Account records and any contacts associated with those accounts. The output returned in SQL isn’t formatted as JSON. Other than that, the SQL and SOQL queries produce the same results.

At this point, you might be wondering whether SOQL supports aliasing. It does, but not how you’re used to working with it in SQL. SOQL has no AS keyword. You can use aliases to represent object names in SOQL queries, but for field names, aliasing works only for aggregate queries, which we’ll be covering next.

Remember, to dive deeper into this topic, check out the hands-on training video link in the Resources section.

What About Aggregates?

Yep, SOQL has aggregates, and they work pretty much the way you expect them to. Well, kind of. The big thing to be aware of when working with aggregates is that for most functions your result is returned as an AggregateResult type.

As far as which functions you can use, SOQL has the ones you see in the table below. See the official docs for more specifics about each function.

SOQL Aggregate Functions

Function Description
AVG() Returns the average value of a numeric field.
COUNT() and COUNT(fieldName) and COUNT_DISTINCT() Returns the number of rows matching the query criteria.
MIN() Returns the minimum value of a field.
MAX() Returns the maximum value of a field.
SUM() Returns the total sum of a numeric field.

To get a record count for a particular table named Account in SQL, you do something like the following:

SELECT COUNT(*) FROM Account

In SOQL, this same query looks like this:

SELECT COUNT() FROM Account

Pretty similar, right?

Where they differ depends on which version of the count function you use, because they return different things. The COUNT() function without a field name is an older version that was available prior to other aggregate functions. It returns an integer and is most similar to the count(*) function offered in SQL.

Count(fieldName) is a newer version that returns the number of rows where the fieldName has a non-null value. What’s different is that it returns that result as a list of AggregateResults and not as a single value.

Let’s see a little of this in action.

  1. In Developer Console, click the Query Editor tab in the bottom pane.
  2. Delete the existing code, and insert the following snippet:
    SELECT COUNT() FROM Account
    
  3. Click Execute. The query results show the total number of rows with a number beside it.
  4. Go back to the Query Editor tab and change the query to be this:
    SELECT COUNT(Id) FROM Account
    
  5. Click Execute. Now the query results show only one row returned and a column that displays the total number of records.

Up until this point, we haven’t talked much about how you handle the data you return from your SOQL queries. But why put off the inevitable? Let’s roll up our sleeves and get down and dirty with some aggregate data.

We’re going to jump into a little bit of Apex, but don’t worry if it doesn’t make complete sense to you yet. We’ll be covering it more in detail later.

  1. In Developer Console, select Debug > Open Execute Anonymous Window.
  2. Delete the existing code, and insert the following snippet:
    List<AggregateResult> results  = [SELECT Industry, count(Id) total
        FROM Account GROUP BY Industry];
    for (AggregateResult ar : results) {
        System.debug('Industry: ' + ar.get('Industry'));
        System.debug('Total Accounts: ' + ar.get('total'));
    }
    

Notice how we used an alias to represent the total along with the GROUP BY clause. In SOQL, you can only alias fields in aggregate queries that use the GROUP BY clause.

  1. Make sure that Open Log is selected, and click Execute. A tab loads showing you the execution log.
  2. Select the Debug Only option so that you see only the debug statements in the log.

Tell Me More

In addition to the GROUP BY clause, SOQL offers other grouping clauses, such as GROUP BY ROLLUP, GROUP BY CUBE, and GROUPING. These clauses are useful for inspecting the results when a query returns values from several related tables. GROUP BY CUBE is a neat little clause that lets you add subtotals for all combinations of a grouped fields in the query results. To learn more about these clauses, look at the GROUP BY document in Resources.

Aggregate functions also support an optional HAVING clause that is similar to the HAVING Clause in SQL Server, so you should feel right at home with this one. It basically lets you filter the results that an aggregate function returns. Check out the Resources to learn more.

Resources

Condividi il tuo feedback su Trailhead dalla Guida di Salesforce.

Conoscere la tua esperienza su Trailhead è importante per noi. Ora puoi accedere al modulo per l'invio di feedback in qualsiasi momento dal sito della Guida di Salesforce.

Scopri di più Continua a condividere il tuo feedback