Mastering LAG and LEAD Functions in SQL Server 2022 with the IGNORE NULLS Option

SQL Server 2022 introduced a powerful enhancement to the LAG and LEAD functions with the IGNORE NULLS option. This feature allows for more precise analysis and reporting by skipping over NULL values in data sets. In this blog, we’ll explore how to use these functions effectively using the JBDB database, and we’ll demonstrate their application with a detailed business use case.

Business Use Case: Sales Data Analysis

Imagine a retail company, JBStore, that wants to analyze its sales data to understand sales trends better. They aim to compare each month’s sales with the previous and next months, ignoring any missing data (represented by NULL values). This analysis will help identify trends and outliers, aiding in better decision-making.

Setting Up the JBDB Database

First, let’s set up the JBDB database and create a SalesData table with some sample data, including NULL values to represent months with no sales data.

-- Create JBDB database
CREATE DATABASE JBDB;
GO

-- Use the JBDB database
USE JBDB;
GO

-- Create SalesData table
CREATE TABLE SalesData (
    SalesMonth INT,
    SalesAmount INT
);

-- Insert sample data, including NULLs
INSERT INTO SalesData (SalesMonth, SalesAmount)
VALUES
    (1, 1000),
    (2, 1500),
    (3, NULL),
    (4, 1800),
    (5, NULL),
    (6, 2000);
GO

LAG and LEAD Functions: A Quick Recap

The LAG function allows you to access data from a previous row in the same result set without the use of a self-join. Similarly, the LEAD function accesses data from a subsequent row. Both functions are part of the SQL window functions family and are particularly useful in time series analysis.

Using LAG and LEAD with IGNORE NULLS

The IGNORE NULLS option is a game-changer, as it allows you to skip over NULL values, providing more meaningful results. Here’s how you can use it with the LAG and LEAD functions:

Example 1: LAG Function with IGNORE NULLS
SELECT 
    SalesMonth,
    SalesAmount,
    LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales
FROM 
    SalesData;

In this example, LAG(SalesAmount, 1) IGNORE NULLS retrieves the sales amount from the previous month, skipping over any NULL values.

Example 2: LEAD Function with IGNORE NULLS
SELECT 
    SalesMonth,
    SalesAmount,
    LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS NextMonthSales
FROM 
    SalesData;

Here, LEAD(SalesAmount, 1) IGNORE NULLS retrieves the sales amount from the next month, again skipping over NULL values.

Practical Example: Analyzing Sales Trends

Let’s combine these functions to analyze sales trends more effectively.

SELECT 
    SalesMonth,
    SalesAmount,
    LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales,
    LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS NextMonthSales
FROM 
    SalesData;

This query provides a complete view of each month’s sales, the previous month’s sales, and the next month’s sales, excluding any NULL values. This is incredibly useful for identifying patterns, such as periods of growth or decline.

Detailed Business Use Case: Data-Driven Decision Making

By utilizing the IGNORE NULLS option with LAG and LEAD functions, JBStore can:

  1. Identify Growth Periods: Detect months where sales increased significantly compared to the previous or next month.
  2. Spot Anomalies: Easily identify months with unusually high or low sales, excluding months with missing data.
  3. Trend Analysis: Understand longer-term trends by comparing sales over multiple months.

These insights can inform marketing strategies, inventory planning, and more.

Calculate Difference Between Current and Previous Month’s Sales:

SELECT SalesMonth, SalesAmount, SalesAmount - LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SalesDifference FROM SalesData;

Identify Months with Sales Decrease Compared to Previous Month:

WITH CTE AS (
    SELECT 
        SalesMonth,
        SalesAmount,
        LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales
    FROM 
        SalesData
)
SELECT 
    SalesMonth,
    SalesAmount,
    PreviousMonthSales
FROM 
    CTE
WHERE 
    SalesAmount < PreviousMonthSales;

Find the Second Previous Month’s Sales:

SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 2) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SecondPreviousMonthSales FROM SalesData;

Calculate the Rolling Average of the Last Two Months (Ignoring NULLs):

SELECT SalesMonth, SalesAmount, (SalesAmount + LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth)) / 2 AS RollingAverage FROM SalesData;

Compare Sales Between Current Month and Two Months Ahead:

SELECT SalesMonth, SalesAmount, LEAD(SalesAmount, 2) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SalesTwoMonthsAhead FROM SalesData;

Identify Consecutive Months with Sales Increase:

WITH CTE AS ( SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales FROM SalesData ) SELECT SalesMonth, SalesAmount FROM CTE WHERE SalesAmount > PreviousMonthSales;

Find Months with No Sales and Their Preceding Sales Month:

SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PrecedingMonthSales FROM SalesData WHERE SalesAmount IS NULL;

Calculate Cumulative Sales Sum Ignoring NULLs:

SELECT 
    SalesMonth,
    SalesAmount,
    SUM(ISNULL(SalesAmount, 0)) OVER (ORDER BY SalesMonth ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS CumulativeSales
FROM 
    SalesData;

Identify the First Month with Sales After a Month with NULL Sales:

SELECT SalesMonth, SalesAmount, LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS FirstNonNullSalesAfterNull FROM SalesData WHERE SalesAmount IS NULL;

    Conclusion 🎉

    The LAG and LEAD functions with the IGNORE NULLS option in SQL Server 2022 offer a more refined way to analyze data, providing more accurate and meaningful results. Whether you’re analyzing sales data, customer behavior, or any other time series data, these functions can significantly enhance your analytical capabilities.

    Happy querying! 🚀

    For more tutorials and tips on SQL Server, including performance tuning and database management, be sure to check out our JBSWiki YouTube channel.

    Thank You,
    Vivek Janakiraman

    Disclaimer:
    The views expressed on this blog are mine alone and do not reflect the views of my company or anyone else. All postings on this blog are provided “AS IS” with no warranties, and confers no rights.

    SQL Server 2022: A Deep Dive into the APPROX_PERCENTILE_CONT Function with JBDB Database

    SQL Server 2022 introduces several new features, one of the most exciting being the APPROX_PERCENTILE_CONT function. This function allows for efficient and approximate calculation of percentiles in large datasets, which can be particularly useful for analytics and data-driven decision-making. In this blog, we will explore the APPROX_PERCENTILE_CONT function in detail, using the JBDB database for practical demonstrations. We’ll start with a business use case, dive into the function’s capabilities, and provide a range of T-SQL queries for you to try. Let’s get started! 🚀


    Business Use Case: Customer Transaction Analysis 💼

    Consider a retail company that wants to analyze customer spending behavior. The company has a vast amount of transaction data stored in the JBDB database. To optimize marketing strategies and tailor promotions, they want to identify spending patterns across different customer segments.

    For example, the company might want to know the 90th percentile of spending amounts to target high-value customers with exclusive offers. Calculating this percentile accurately in a large dataset can be resource-intensive. The APPROX_PERCENTILE_CONT function offers a solution by providing an approximate, yet efficient, calculation of percentiles.


    Understanding the APPROX_PERCENTILE_CONT Function 📊

    The APPROX_PERCENTILE_CONT function is designed to compute approximate percentile values for a set of data. This function is particularly useful when dealing with large datasets, as it offers a performance advantage by using approximate algorithms.

    Syntax:

    APPROX_PERCENTILE_CONT ( percentile ) WITHIN GROUP ( ORDER BY numeric_expression )
    • percentile: A value between 0 and 1 that specifies the desired percentile.
    • numeric_expression: The column or expression to calculate the percentile on.

    Example 1: Basic Usage 🌟

    Let’s calculate the 90th percentile of customer transaction amounts.

    Setup:

    USE JBDB;
    GO
    
    CREATE TABLE CustomerTransactions (
        TransactionID INT PRIMARY KEY,
        CustomerID INT,
        TransactionAmount DECIMAL(18, 2),
        TransactionDate DATE
    );
    
    INSERT INTO CustomerTransactions (TransactionID, CustomerID, TransactionAmount, TransactionDate)
    VALUES
    (1, 101, 50.00, '2023-01-15'),
    (2, 102, 150.00, '2023-01-16'),
    (3, 103, 300.00, '2023-01-17'),
    (4, 101, 75.00, '2023-01-18'),
    (5, 104, 200.00, '2023-01-19'),
    (6, 105, 125.00, '2023-01-20'),
    (7, 106, 400.00, '2023-01-21'),
    (8, 102, 175.00, '2023-01-22');
    GO

    Query to Calculate 90th Percentile:

    SELECT APPROX_PERCENTILE_CONT(0.90) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx90thPercentile
    FROM CustomerTransactions;

    This result indicates that 90% of transactions are below $375. This insight can help the company focus on high-value customers who spend above this threshold.

    Example 2: Analyzing Different Percentiles 🔍

    Let’s calculate different percentiles to understand the distribution of transaction amounts.

    Query to Calculate Multiple Percentiles:

    SELECT 
        APPROX_PERCENTILE_CONT(0.25) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx25thPercentile,
        APPROX_PERCENTILE_CONT(0.50) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx50thPercentile,
        APPROX_PERCENTILE_CONT(0.75) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx75thPercentile,
        APPROX_PERCENTILE_CONT(0.90) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx90thPercentile
    FROM CustomerTransactions;

    These results provide a clear view of the transaction distribution, helping the company to tailor marketing strategies for different customer segments.

    Comparing Percentile Results:

    • Compare approximate and exact percentile calculations for the 90th percentile:
    SELECT 
        APPROX_PERCENTILE_CONT(0.90) WITHIN GROUP (ORDER BY TransactionAmount) AS Approx90thPercentile,
        PERCENTILE_CONT(0.90) WITHIN GROUP (ORDER BY TransactionAmount) OVER () AS Exact90thPercentile
    FROM CustomerTransactions
    group by TransactionAmount;
    

    Segmenting Customers by Spending:

    • Identify customers whose spending is in the top 10%:
    SELECT CustomerID, TransactionAmount
    FROM CustomerTransactions
    WHERE TransactionAmount >= (SELECT APPROX_PERCENTILE_CONT(0.90) WITHIN GROUP (ORDER BY TransactionAmount)
                                 FROM CustomerTransactions);

    Analyzing Spending Patterns Over Time:

    • Calculate monthly spending percentiles to identify trends:
    SELECT 
        DATEPART(MONTH, TransactionDate) AS Month,
        APPROX_PERCENTILE_CONT(0.50) WITHIN GROUP (ORDER BY TransactionAmount) AS MedianTransaction
    FROM CustomerTransactions
    GROUP BY DATEPART(MONTH, TransactionDate)
    ORDER BY Month;
    

    Combining Percentiles with Other Aggregations:

    • Find the average transaction amount for each percentile group:
    SELECT 
        PercentileGroup,
        AVG(TransactionAmount) AS AvgTransactionAmount
    FROM (
        SELECT 
            TransactionAmount,
            NTILE(4) OVER (ORDER BY TransactionAmount) AS PercentileGroup
        FROM CustomerTransactions
    ) AS SubQuery
    GROUP BY PercentileGroup;
    

    Conclusion 🏁

    The APPROX_PERCENTILE_CONT function in SQL Server 2022 is a powerful tool for efficiently computing approximate percentiles in large datasets. By using this function, businesses can gain valuable insights into data distributions and make informed decisions based on these insights. Whether you’re analyzing customer spending, sales trends, or any other data, the APPROX_PERCENTILE_CONT function offers a quick and efficient way to understand your data.

    Happy querying! 😄

    For more tutorials and tips on SQL Server, including performance tuning and database management, be sure to check out our JBSWiki YouTube channel.

    Thank You,
    Vivek Janakiraman

    Disclaimer:
    The views expressed on this blog are mine alone and do not reflect the views of my company or anyone else. All postings on this blog are provided “AS IS” with no warranties, and confers no rights.