This is legacy documentation, and may not be complete. To see the latest documentation, if you are a marketer, refer to the Marketers site. If you are a measurement partner, refer to the Measurement Partners site.

Best practices

  • Utilize sandbox data whenever possible for query development to avoid privacy checks and improve speed.

  • Minimize the likelihood of result set overlap between recently run queries by modifying key parameters.

  • Avoid running multiple queries where the end date is today, as this often leads to filtered rows.

  • Limit the number of times you query the same data by using tightly bound dates, disjoint sets, saved results, or temp tables.

  • Minimize the number of cross-user aggregations and rewrite queries to combine aggregations or split them into more focused queries.

The following best practices will provide you with techniques to develop privacy-centric and performant queries. For best practices specific to running queries in noise mode, see the sections on supported and unsupported query patterns in Noise injection.

Privacy and data accuracy

Develop queries on sandbox data

Best practice: Only query production data when you're in production.

Utilize sandbox data during your query development whenever possible. Jobs using sandbox data don't introduce additional opportunities for difference checks to filter your query results. Additionally, because of the lack of privacy checks, sandbox queries run faster, allowing for more rapid iteration during query development.

If you have to develop queries on your actual data (such as when using match tables), to make it less likely that you overlap rows, choose date ranges and other parameters which are unlikely to overlap for each iteration of your query. Finally, run your query over the desired range of data.

Carefully consider historical results

Best practice: Decrease the likelihood of result set overlap between recently run queries.

Bear in mind that the rate of change between query results will have an effect on how likely it is that results are omitted later on due to privacy checks. A second result set that closely resembles a recently returned result set is likely to be dropped.

Instead, modify key parameters in your query, such as date ranges or campaign IDs, to decrease the likelihood of significant overlap.

Don't query today's data

Best practice: Don't run multiple queries where the end date is today.

Running multiple queries with end dates equal to today will often lead to rows being filtered. This guidance also applies to running queries shortly after midnight on yesterday's data.

Don't query the same data more than necessary

Best practices:

  • Select tightly bound start and end dates.
  • Instead of querying overlapping windows, run your queries on disjoint sets of data, then aggregate the results in BigQuery.
  • Use saved results instead of re-running your query.
  • Create temp tables for each date range you're querying over.

Ads Data Hub restricts the total number of times that you can query the same data. As such, you should attempt to limit the number of times that you access a given piece of data.

Don't use more aggregations than necessary in the same query

Best practices:

  • Minimize the number of aggregations in a query
  • Rewrite queries to combine aggregations when possible

Ads Data Hub limits the number of cross-user aggregations allowed to be used in a subquery to 100. Hence, overall we recommend writing queries that output more rows with focused grouping keys and simple aggregations, rather than more columns with broad grouping keys and complex aggregations. The following pattern should be avoided:

SELECT
COUNTIF(field_1=a_1ANDfield_2=b_1)AScnt_1,
COUNTIF(field_1=a_2ANDfield_2=b_2)AScnt_2
FROM
table

Queries that count events depending on the same set of fields should be rewritten using the GROUP BY statement.

SELECT
field_1,
field_2,
COUNT(1)AScnt
FROM
table
GROUPBY
1,2

The result can be aggregated the same way in BigQuery.

Queries that create columns from an array and then aggregate them afterwards should be rewritten to merge these steps.

SELECT
COUNTIF(a_1)AScnt_1,
COUNTIF(a_2)AScnt_2
FROM
(SELECT
1INUNNEST(field)ASa_1,
2INUNNEST(field)ASa_2,
FROM
table)

The previous query can be rewritten as:

SELECTf,COUNT(1)FROMtable,UNNEST(field)ASfGROUPBY1

Queries that use different combinations of fields in different aggregations can be rewritten into several more focused queries.

SELECT
COUNTIF(field_1=a_1)AScnt_a_1,
COUNTIF(field_1=b_1)AScnt_b_1,
COUNTIF(field_2=a_2)AScnt_a_2,
COUNTIF(field_2=b_2)AScnt_b_2,
FROMtable

The previous query can be split into:

SELECT
field_1,COUNT(*)AScnt
FROMtable
GROUPBY1

and

SELECT
field_2,COUNT(*)AScnt
FROMtable
GROUPBY1

You can split these results into separate queries, create and join the tables in a single query, or combine them with a UNION if schemas are compatible.

Optimize and understand joins

Best practice: Use a LEFT JOIN instead of an INNER JOIN to join clicks or conversions to impressions.

Not all impressions are associated with clicks or conversions. Therefore, if you INNER JOIN clicks or conversions on impressions, impressions that aren't tied to clicks or conversions will be filtered from your results.

Image showing multiple join types via venn diagrams

Join some final results in BigQuery

Best practice: Avoid Ads Data Hub queries that join aggregated results. Instead, write 2 separate queries and join the results in BigQuery.

Rows that don't meet aggregation requirements are filtered from your results. Therefore, if your query joins an insufficiently aggregated row with a sufficiently aggregated row, the resultant row will be filtered. Additionally, queries with multiple aggregations are less performant in Ads Data Hub.

You can join results (in BigQuery) from multiple aggregation queries (from Ads Data Hub). Results computed using common queries will share final schemas.

The following query takes individual Ads Data Hub results (campaign_data_123 and campaign_data_456) and joins them in BigQuery:

SELECTt1.campaign_id,t1.city,t1.X,t2.Y
FROM`campaign_data_123`ASt1
FULLJOIN`campaign_data_456`ASt2
USING(campaign_id,city)

Use filtered row summaries

Best practice: Add filtered row summaries to your queries.

Filtered row summaries tally data that was filtered due to privacy checks. Data from filtered rows is summed and added to a catch-all row. While the filtered data can't be further analyzed, it provides a summary of how much data was filtered from the results.

Account for zeroed user IDs

Best practice: Account for zeroed user IDs in your results.

An end-user's ID may be set to 0 for a number of reasons, including: opting out of ads personalization, regulatory reasons, etc. As such, data originating from multiple users will be keyed to a user_id of 0.

If you want to understand data totals, such as total impressions or clicks, you should include these events. However, this data won't be useful for deriving insights on customers, and should be filtered if you're doing such analysis.

You can exclude this data from your results by adding WHERE user_id != "0" to your queries.


Performance

Avoid reaggregation

Best practice: Avoid multiple layers of aggregation across users.

Queries that combine results that have already been aggregated, such as in the case of a query with multiple GROUP BYs, or nested aggregation, require more resources to process.

Often, queries with multiple layers of aggregation can be broken up, improving performance. You should attempt to keep rows at the event or user level while processing and then combine with a single aggregation.

The following patterns should be avoided:

SELECTSUM(count)
FROM
(SELECTcampaign_id,COUNT(0)AScountFROM...GROUPBY1)

Queries that use multiple layers of aggregation should be rewritten to use a single layer of aggregation.

(SELECT...GROUPBY...)
JOINUSING(...)
(SELECT...GROUPBY...)

Queries that can easily be broken up should be broken up. You can join results in BigQuery.

Optimize for BigQuery

Generally, queries that do less perform better. When evaluating query performance, the amount of work required depends on the following factors:

If query execution isn't meeting your service level agreements, or you're encountering errors due to resource exhaustion or timeout, consider:

  • Using the results from previous queries instead of recomputing. For example, your weekly total could be the sum computed in BigQuery of 7 single day aggregate queries.
  • Decomposing queries into logical subqueries (such as splitting multiple joins into multiple queries), or otherwise restricting the set of data being processed. You can combine results from individual jobs into a single dataset in BigQuery. Although this may help with resource exhaustion, it may slow down your query.
  • If you're running into resources exceeded errors in BigQuery, try using temp tables to split your query into multiple BigQuery queries.
  • Referencing fewer tables in a single query, as this uses large amounts of memory and can cause your query to fail.
  • Rewriting your queries such that they join fewer user tables.
  • Rewriting your queries to avoid joining the same table back on itself.

Query advisor

If your SQL is valid but might trigger excessive filtering, the query advisor surfaces actionable advice during the query development process, to help you avoid undesirable results.

Triggers include the following patterns:

To use the query advisor:

Except as otherwise noted, the content of this page is licensed under the Creative Commons Attribution 4.0 License, and code samples are licensed under the Apache 2.0 License. For details, see the Google Developers Site Policies. Java is a registered trademark of Oracle and/or its affiliates.

Last updated 2025年10月22日 UTC.