- Categories:

# MATCH_RECOGNIZE¶

Recognizes matches of a pattern in a set of rows. `MATCH_RECOGNIZE`

accepts a set of rows (from a table,
view, subquery, or other source) as input, and returns all matches for a given row pattern within this
set. The pattern is defined similarly to a regular expression.

The clause can return either:

All the rows belonging to each match.

One summary row per match.

`MATCH_RECOGNIZE`

is typically used to detect events in time series. For example, `MATCH_RECOGNIZE`

can search a
stock price history table for shapes like `V`

(down followed by up) or `W`

(down, up, down, up).

`MATCH_RECOGNIZE`

is an optional subclause of the FROM clause.

## Syntax¶

```
MATCH_RECOGNIZE (
[ PARTITION BY <expr> [, ... ] ]
[ ORDER BY <expr> [, ... ] ]
[ MEASURES <expr> [AS] <alias> [, ... ] ]
[ ONE ROW PER MATCH |
ALL ROWS PER MATCH [ { SHOW EMPTY MATCHES | OMIT EMPTY MATCHES | WITH UNMATCHED ROWS } ]
]
[ AFTER MATCH SKIP
{
PAST LAST ROW |
TO NEXT ROW |
TO [ { FIRST | LAST} ] <symbol>
}
]
PATTERN ( <pattern> )
DEFINE <symbol> AS <expr> [, ... ]
)
```

## Required Subclauses¶

### DEFINE: Defining Symbols¶

```
DEFINE <symbol1> AS <expr1> [ , <symbol2> AS <expr2> ]
```

Symbols (also known as “pattern variables”) are the building blocks of the pattern.

A symbol is defined by an expression. If the expression evaluates to true for a row, the symbol is assigned to that row. A row can be assigned multiple symbols.

Symbols that are not defined in the `DEFINE`

clause, but are used in the pattern, are always assigned to all
rows. Implicitly, they are equivalent to the following example:

```
...
define
my_example_symbol as true
...
```

Patterns are defined based on symbols and operators.

### PATTERN: Specifying the Pattern to Match¶

```
PATTERN ( <pattern> )
```

The pattern defines a valid sequence of rows that represents a match. The pattern is defined like a regular expression (regex) and is built from symbols, operators, and quantifiers.

For example, suppose that symbol `S1`

is defined as `stock_price < 55`

, and symbol `S2`

is defined
as `stock price > 55`

. The following pattern specifies a sequence of rows in which the stock price increased
from less than 55 to greater than 55:

```
PATTERN (S1 S2)
```

The following is a more complex example for a pattern definition:

```
^ S1 S2*? ( {- S3 -} S4 )+ | PERMUTE(S2 S1, S4){1,2} $
```

The following section describes the individual components of this pattern in detail.

- Symbols

A symbol matches to a row that symbol was assigned to. The following symbols are available:

. For example,*symbol*`S1`

, … ,`S4`

Those are symbols that were defined in the`DEFINE`

subclause and are evaluated per row. (These can also include symbols that were not defined and are automatically assigned to all rows.)

(Start of partition.) This is a virtual symbol that denotes the start of a partition and has no row associated with it. You can use it to require a match to start only at the beginning of a partition.*^*For an example, see Matching Patterns Relative to the Beginning or End of a Partition.

(End of partition.) This is a virtual symbol that denotes the end of a partition and has no row associated with it. You can use it to require a match to end only at the end of a partition.*$*For an example, see Matching Patterns Relative to the Beginning or End of a Partition.

- Quantifiers

A quantifier can be placed following a symbol or operation. A quantifier denotes the minimum and maximum number of occurrences of the associated symbol or operation. The following quantifiers are available:

Quantifier

Meaning

`+`

1 or more. For example,

`( {- S3 -} S4 )+`

.

`*`

0 or more. For example,

`S2*?`

.

`?`

0 or 1.

`{n}`

Exactly n.

`{n,}`

n or more.

`{,m}`

0 to m.

`{n, m}`

n to m. For example,

`PERMUTE(S2 S1, S4){1,2}`

.

By default, quantifiers are in “greedy mode”, which means they try to match the maximum quantity if possible. To put a
quantifier into “reluctant mode”, in which the quantifier tries to match the minimum quantity if possible,
place a `?`

after the quantifier (e.g. `S2*?`

).

- Operators

Operators specify in which order symbols or other operations should occur in the sequence of rows to form a valid match. The following operators are available:

Operator

Meaning

`... ...`

(space)Concatenation. Specifies that a symbol or operation should follow another one. For example,

`S1 S2`

means that the condition defined for`S2`

should occur after the condition defined for`S1`

.

`{- ... -}`

Exclusion. Excludes the contained symbols or operations from the output. For example,

`{- S3 -}`

excludes operator`S3`

from the output. Excluded rows will not appear in the output, but will be included in the evaluation of`MEASURES`

expressions.

`( ... )`

Grouping. Used to override the precedence of an operator or to apply the same quantifier for symbols or operations in the group. In this example, the quantifier

`+`

applies to the sequence`{- S3 -} S4`

, not merely`S4`

.

`PERMUTE( ..., ...)`

Permutation. Specifies that any permutation of the arguments is allowed. For example,

`PERMUTE(S2 S1, S4)`

means that either`S4`

or`S2 S1`

matches the pattern.`PERMUTE`

takes an unlimited number of arguments.

`... | ...`

Alternative. Specifies that either the first symbol or operation or the other one should occur. For example,

`( S3 S4 ) | PERMUTE(S2 S1, S4)`

. The alternative operator has precedence over the concatenation operator.

## Optional Subclauses¶

### ORDER BY: Sorting the Rows Before Matching¶

*ORDER BY orderItem1 [ , orderItem2 … ]*

Where:

orderItem ::= { <column_alias> | <expr> } [ ASC | DESC ] [ NULLS { FIRST | LAST } ]Define the order of the rows as you would for window functions. This is the order in which the individual rows of each partition are passed to the

`MATCH_RECOGNIZE`

operator.For more information, see Partitioning and Sorting the Rows.

### PARTITION BY: Partitioning The Rows Into Windows¶

*PARTITION BY <expr1> [ , <expr2> … ]*

Partition the input set of rows as you would for window functions.

`MATCH_RECOGNIZE`

performs matching individually for each resulting partition.Partitioning not only groups rows that are related to each other, but also leverages Snowflake’s distributed data processing capability because separate partitions can be processed in parallel.

For more information about partitioning, see Partitioning and Sorting the Rows.

### MEASURES: Specifying Additional Output Columns¶

```
MEASURES <expr1> [AS] <alias1> [ ... , <exprN> [AS] <aliasN> ]
```

“Measures” are optional additional columns that are added to the output of the `MATCH_RECOGNIZE`

operator.
The expressions in the `MEASURES`

subclause have the same capabilities as the expressions in the `DEFINE`

subclause. For further information, see Symbols.

Within the `MEASURES`

subclause, the following functions specific to `MATCH_RECOGNIZE`

are available:

`MATCH_NUMBER()`

Returns the sequential number of the match. The MATCH_NUMBER starts from 1, and is incremented for each match.`MATCH_SEQUENCE_NUMBER()`

Returns the row number within a match. The MATCH_SEQUENCE_NUMBER is sequential and starts from 1.`CLASSIFIER()`

Returns a TEXT value that contains the symbol that the respective row matched. For example, if a row matched the symbol`GT75`

, then the`CLASSIFIER`

function returns the string “GT75”.

Note

When specifying measures, remember the restrictions mentioned in the Limitations on Window Functions used in DEFINE and MEASURES section.

### ROW(S) PER MATCH: Specifying the Rows to Return¶

```
{
ONE ROW PER MATCH |
ALL ROWS PER MATCH [ { SHOW EMPTY MATCHES | OMIT EMPTY MATCHES | WITH UNMATCHED ROWS } ]
}
```

Specifies which rows are returned for a successful match. This subclause is optional.

`ALL ROWS PER MATCH`

: Return all rows in the match.`ONE ROW PER MATCH`

: Return one summary row for each match, regardless of how many rows are in the match. This is the default.

Be aware of the following special cases:

Empty Matches: An empty match happens if a pattern is able to match against zero rows. For instance, if the pattern is defined as

`A*`

and the first row at the beginning of a matching attempt is assigned to symbol`B`

, then an empty match including only that row is generated, because the`*`

quantifier in the`A*`

pattern allows 0 occurrences of`A`

to be treated as a match. The`MEASURES`

expressions are evaluated differently for this row:The CLASSIFIER function returns NULL.

Window functions return NULL.

The COUNT function returns 0.

Unmatched Rows: If a row was not matched against the pattern, it is called an unmatched row.

`MATCH_RECOGNIZE`

can be configured to return unmatched rows, too. For unmatched rows, expressions in the`MEASURES`

subclause return NULL.

Exclusions

The exclusion syntax

`({- ... -})`

in the pattern definition allows users to exclude certain rows from the output. If all matched symbols in the pattern were excluded, no row is generated for that match if`ALL ROWS PER MATCH`

was specified. Note that the MATCH_NUMBER is incremented anyway. Excluded rows are not part of the result, but are included for the evaluation of`MEASURES`

expressions.When using the exclusion syntax, the ROWS PER MATCH subclause can be specified as follows:

ONE ROW PER MATCH (default)

Returns exactly one row for each successful match. The default window function semantic for window functions in the

`MEASURES`

subclause is`FINAL`

.The output columns of the

`MATCH_RECOGNIZE`

operator are all expressions given in the`PARTITION BY`

subclause and all`MEASURES`

expressions. All resulting rows of a match are grouped by the expressions given in the`PARTITION BY`

subclause and the`MATCH_NUMBER`

using the`ANY_VALUE`

aggregation function for all measures. Therefore, if measures evaluate to a different value for different rows of the same match, then the output is non-deterministic.Omitting the

`PARTITION BY`

and`MEASURES`

subclause results in an error indicating that the result does not include any columns.For empty matches, a row is generated. Unmatched rows are not part of the output.

`ALL ROWS PER MATCH`

Returns a row for each row that is part of the match, except for rows that were matched to a portion of the pattern that was marked for exclusion.

Excluded rows are still taken into account in computations in the

`MEASURES`

subclause.Matches might overlap based on the

`AFTER MATCH SKIP TO`

subclause, so the same row might appear multiple times in the output.The default window function semantic for window functions in the

`MEASURES`

subclause is`RUNNING`

.The output columns of the

`MATCH_RECOGNIZE`

operator are the columns of the set of rows being input and the columns defined in the`MEASURES`

subclause.The following options are available for

`ALL ROWS PER MATCH`

:`SHOW EMPTY MATCHES (default)`

Add empty matches to the output. Unmatched rows are not output.`OMIT EMPTY MATCHES`

Neither empty matches nor unmatched rows are output. However, the MATCH_NUMBER is still incremented by an empty match.`WITH UNMATCHED ROWS`

Adds empty matches and unmatched rows to the output. If this clause is used, then the pattern must not contain exclusions.

For an example that uses exclusion to reduce irrelevant output, see Search for Patterns in Non-Adjacent Rows.

### AFTER MATCH SKIP: Specifying Where to Continue After a Match¶

```
AFTER MATCH SKIP
{
PAST LAST ROW |
TO NEXT ROW |
TO [ { FIRST | LAST} ] <symbol>
}
```

This subclause specifies where to continue the matching after a positive match was found.

`PAST LAST ROW (default)`

Continue matching after the last row of the current match.

This prevents matches that contain overlapping rows. For example, if you have a stock pattern that contains 3

`V`

shapes in a row, then`PAST LAST ROW`

finds one`W`

pattern, not two.`TO NEXT ROW`

Continue matching after the first row of the current match.

This allows matches that contain overlapping rows. For example, if you have a stock pattern that contains 3

`V`

shapes in a row, then`TO NEXT ROW`

finds two`W`

patterns (the first pattern is based on the first two`V`

shapes, and the second`W`

shape is based on the second and third`V`

shapes; thus both patterns contain the same`V`

).`TO [ { FIRST | LAST } ] <symbol>`

Continue matching at the first or last (default) row that was matched to the given symbol.

At least one row needs to be mapped to the given symbol or an error is raised.

If this does not skip past the first row of the current match, then an error is raised.

## Usage Notes¶

### Expressions in DEFINE and MEASURES clauses¶

The `DEFINE`

and `MEASURES`

clauses allow expressions. Those expressions can be complex and can include
window functions and special navigational functions (which are a type of
window function).

In most respects, expressions in `DEFINE`

and `MEASURES`

follow the rules for expressions elsewhere in Snowflake
SQL syntax. However, there are some differences, which are described below:

- Window Functions
`PREV(<expr> [, <offset> [, <default> ] ])`

Navigate to the previous row within the current window frame. This function is similar to`LAG`

, but for`PREV()`

the window frame is limited to the beginning of the current matching attempt when`PREV()`

is used within the`DEFINE`

subclause.For example, you might define a drop in price as shown below:

define down as prev(price) > price

This expression is true when the price in the current row (e.g. today) is lower than the price in the previous row (e.g. yesterday).

`NEXT(<expr> [, <offset> [, <default> ] ])`

Navigate to the next row within the current window frame. This function is equivalent to`LEAD`

.`FIRST(<expr>)`

Navigate to the first row of the current window frame. This function is similar to`FIRST_VALUE`

, but for`FIRST()`

the window frame is limited to the beginning of the current matching attempt when`FIRST`

is used within the`DEFINE`

subclause.`LAST(<expr>)`

Navigate to the last row of the current window frame. This function is similar to`LAST_VALUE`

, but for`LAST()`

the window frame is limited to the current row of the current matching attempt when`LAST()`

is used within the`DEFINE`

subclause.

For an example that uses the navigational functions, see Returning Information About the Match.

In general, when a window function is used inside a

`MATCH_RECOGNIZE`

clause, the window function does not require its own`OVER (PARTITION BY ... ORDER BY ...)`

clause. The window is implicitly determined by the`PARTITION BY`

and`ORDER BY`

of the`MATCH_RECOGNIZE`

clause. (However, see Limitations on Window Functions used in DEFINE and MEASURES for some exceptions.)In general, the window frame is also derived implicitly from the current context in which the window function is being used. The lower bound of the frame is defined as described below:

In the

`DEFINE`

subclause:The frame starts at the beginning of the current matching attempt except when using

`LAG`

,`LEAD`

,`FIRST_VALUE`

, and`LAST_VALUE`

.In the

`MEASURES`

subclause:The frame starts at the beginning of the match that was found.

The edges of the window frame can be specified by using either

`RUNNING`

or`FINAL`

semantics.expr ::= ... [ { RUNNING | FINAL } ] windowFunction ...

`RUNNING`

:In general, the frame ends at the current row. However, the following exceptions exist:

In the

`DEFINE`

subclause, for`LAG`

,`LEAD`

,`FIRST_VALUE`

,`LAST_VALUE`

,`NEXT`

, and`PREV`

, the frame ends at the last row of the window.In the

`MEASURES`

subclause, for`PREV`

,`NEXT`

,`LAG`

, and`LEAD`

, the frame ends at the last row of the window.

In the

`DEFINE`

subclause,`RUNNING`

is the default (and the only allowed) semantic.In the

`MEASURES`

subclause, when the`ALL ROWS PER MATCH`

subclause is used,`RUNNING`

is the default.`FINAL`

:The frame ends at the last row of the match.

`FINAL`

is allowed only in the`MEASURES`

subclause. It is the default there when`ONE ROW PER MATCH`

applies.- Symbol Predicates
Expressions within the

`DEFINE`

and`MEASURES`

subclauses allow symbols as predicates for column references.predicatedColumnReference ::= <symbol>.<column>

A predicated column reference means that the surrounding window function only looks at rows that were finally mapped to the specified symbol.

Predicated column references can be used outside and inside of a window function. If used outside of a window function,

is the same as*<symbol>.<column>*`LAST(<symbol>.<column>)`

. Inside of a window function, all column references either need to be predicated with the same symbol or are all non-predicated.The following explains how navigational-related functions behave with predicated column references:

Searches the window frame backwards starting from and including the current row (or last row in case of a*PREV/LAG( … <symbol>.<column> … , <offset>)*`FINAL`

semantic) for the first row that was finally mapped to the specified`<symbol>`

and goes`<offset>`

(default is 1) rows backwards ignoring the symbol those rows were mapped to. If the searched part of the frame does not contain a row mapped to <symbol> or the frame is left, then NULL is returned.

Searches the window frame backwards starting from and including the current row (or last row in case of a*NEXT/LEAD( … <symbol>.<column> … , <offset>)*`FINAL`

semantic) for the first row that was finally mapped to the specified <symbol> and goes <offset> (default is 1) rows forward ignoring the symbol those rows were mapped to. If the searched part of the frame does not contain a row mapped to <symbol> or the frame is left, then NULL is returned.

Searches the window frame forwards starting from and including the first row up to and including the current row (or last row in case of a*FIRST/FRST_VALUE( … <symbol>.<column> … )*`FINAL`

semantic) for the first row that was finally mapped to the specified <symbol>. If the searched part of the frame does not contain a row mapped to <symbol>, NULL is returned.

Searches the window frame backwards starting from and including the current row (or last row in case of a*LAST/LAST_VALUE( … <symbol>.<column> … )*`FINAL`

semantic) for the first row that was finally mapped to the specified <symbol>. If the searched part of the frame does not contain a row mapped to <symbol>, NULL is returned.

Note

Restrictions on window functions are documented in the Limitations on Window Functions used in DEFINE and MEASURES section.

### Limitations on Window Functions used in DEFINE and MEASURES¶

Expressions in the `DEFINE`

and `MEASURES`

subclauses can include window functions. However, there are some
limitations on using window functions in these subclauses. These limitations are shown in the table below:

Function

DEFINE (Running) [column/symbol.column]

MEASURES (Running) [column/symbol.column]

MEASURES (Final) [column/symbol.column]

Column

✔ / ❌

✔ / ❌

✔ / ✔

PREV(…)

❌ / ❌

✔ / ❌

✔ / ❌

NEXT(…)

✔ / ❌

✔ / ❌

✔ / ❌

FIRST(…)

❌ / ❌

✔ / ❌

✔ / ✔

LAST(…)

✔ / ❌

✔ / ❌

✔ / ✔

LAG()

✔ / ❌

✔ / ❌

✔ / ❌

LEAD()

✔ / ❌

✔ / ❌

✔ / ❌

FIRST_VALUE()

✔ / ❌

✔ / ❌

✔ / ✔

LAST_VALUE()

✔ / ❌

✔ / ❌

✔ / ✔

Aggregations 1

✔ / ❌

✔ / ✔

✔ / ✔

Other window functions 1

✔ / ❌

✔ / ❌

✔ / ❌

- 1(1,2)
These functions require an explicit frame definition

`(OVER (ROWS BETWEEN ...))`

when used in the`DEFINE`

clause.

The `MATCH_RECOGNIZE`

-specific functions `MATCH_NUMBER()`

, `MATCH_SEQUENCE_NUMBER()`

, and `CLASSIFIER()`

are
currently not available in the `DEFINE`

subclause.

## Troubleshooting¶

### Error Message: `SELECT with no columns`

When Using `ONE ROW PER MATCH`

¶

When you use the `ONE ROW PER MATCH`

clause, only columns and expressions from the `PARTITION BY`

and `MEASURES`

subclauses are allowed in the projection clause of the SELECT. If you try to use `MATCH_RECOGNIZE`

without either a
`PARTITION BY`

or `MEASURES`

clause, you get an error similar to `SELECT with no columns`

.

For more information about `ONE ROW PER MATCH`

vs. `ALL ROWS PER MATCH`

,
see Generating One Row for Each Match vs. Generating All Rows for Each Match.

## Examples¶

The topic Identifying Sequences of Rows That Match a Pattern contains many examples, including some that are simpler than
most of the examples here. If you are not already familiar with `MATCH_RECOGNIZE`

, then you might want to read those
examples first.

Some of the examples below use the following table and data:

create table stock_price_history (company TEXT, price_date DATE, price INT);insert into stock_price_history values ('ABCD', '2020-10-01', 50), ('XYZ' , '2020-10-01', 89), ('ABCD', '2020-10-02', 36), ('XYZ' , '2020-10-02', 24), ('ABCD', '2020-10-03', 39), ('XYZ' , '2020-10-03', 37), ('ABCD', '2020-10-04', 42), ('XYZ' , '2020-10-04', 63), ('ABCD', '2020-10-05', 30), ('XYZ' , '2020-10-05', 65), ('ABCD', '2020-10-06', 47), ('XYZ' , '2020-10-06', 56), ('ABCD', '2020-10-07', 71), ('XYZ' , '2020-10-07', 50), ('ABCD', '2020-10-08', 80), ('XYZ' , '2020-10-08', 54), ('ABCD', '2020-10-09', 75), ('XYZ' , '2020-10-09', 30), ('ABCD', '2020-10-10', 63), ('XYZ' , '2020-10-10', 32);

The following graph shows the shapes of the curves:

### Report One Summary Row for Each `V`

Shape¶

The following query searches for all `V`

shapes in the previously presented stock_price_history. The output is
explained in more detail after the query and output.

SELECT * FROM stock_price_history MATCH_RECOGNIZE( PARTITION BY company ORDER BY price_date MEASURES MATCH_NUMBER() AS match_number, FIRST(price_date) AS start_date, LAST(price_date) AS end_date, COUNT(*) AS rows_in_sequence, COUNT(row_with_price_decrease.*) AS num_decreases, COUNT(row_with_price_increase.*) AS num_increases ONE ROW PER MATCH AFTER MATCH SKIP TO LAST row_with_price_increase PATTERN(row_before_decrease row_with_price_decrease+ row_with_price_increase+) DEFINE row_with_price_decrease AS price < LAG(price), row_with_price_increase AS price > LAG(price) ) ORDER BY company, match_number; +---------+--------------+------------+------------+------------------+---------------+---------------+ | COMPANY | MATCH_NUMBER | START_DATE | END_DATE | ROWS_IN_SEQUENCE | NUM_DECREASES | NUM_INCREASES | |---------+--------------+------------+------------+------------------+---------------+---------------| | ABCD | 1 | 2020-10-01 | 2020-10-04 | 4 | 1 | 2 | | ABCD | 2 | 2020-10-04 | 2020-10-08 | 5 | 1 | 3 | | XYZ | 1 | 2020-10-01 | 2020-10-05 | 5 | 1 | 3 | | XYZ | 2 | 2020-10-05 | 2020-10-08 | 4 | 2 | 1 | | XYZ | 3 | 2020-10-08 | 2020-10-10 | 3 | 1 | 1 | +---------+--------------+------------+------------+------------------+---------------+---------------+

The output shows one row per match (regardless of how many rows were part of the match).

The output includes the following columns:

COMPANY: The stock symbol for the company.

The MATCH_NUMBER is a sequential number identifying which match this was within this data set (e.g. the first match has MATCH_NUMBER 1, the second match has MATCH_NUMBER 2, etc.). If the data was partitioned, then the MATCH_NUMBER is the sequential number within the partition (in this example, for each company/stock).

START_DATE: The date at which this occurrence of the pattern starts.

END_DATE: The date at which this occurrence of the pattern ends.

ROWS_IN_SEQUENCE: This is the number of rows in the match. For example, the first match is based on the prices measured on 4 days (October 1 through October 4), so ROWS_IN_SEQUENCE is 4.

NUM_DECREASES: This is the number of days (within the match) that the price went down. For example, in the first match, the price went down for 1 day and then went up for 2 days, so NUM_DECREASES is 1.

NUM_INCREASES: This is the number of days (within the match) that the price went up. For example, in the first match, the price went down for 1 day and then went up for 2 days, so NUM_INCREASES is 2.

### Report All Rows for All Matches for One Company¶

This example returns all rows within each match (not just one summary row per match). This pattern searches for rising prices of the ‘ABCD’ company:

select price_date, match_number, msq, price, cl from (select * from stock_price_history where company='ABCD') match_recognize( order by price_date measures match_number() as "MATCH_NUMBER", match_sequence_number() as msq, classifier() as cl all rows per match pattern(ANY_ROW UP+) define ANY_ROW AS TRUE, UP as price > lag(price) ) order by match_number, msq; +------------+--------------+-----+-------+---------+ | PRICE_DATE | MATCH_NUMBER | MSQ | PRICE | CL | |------------+--------------+-----+-------+---------| | 2020-10-02 | 1 | 1 | 36 | ANY_ROW | | 2020-10-03 | 1 | 2 | 39 | UP | | 2020-10-04 | 1 | 3 | 42 | UP | | 2020-10-05 | 2 | 1 | 30 | ANY_ROW | | 2020-10-06 | 2 | 2 | 47 | UP | | 2020-10-07 | 2 | 3 | 71 | UP | | 2020-10-08 | 2 | 4 | 80 | UP | +------------+--------------+-----+-------+---------+

### Omit Empty Matches¶

This searches for price ranges above the average of the whole chart of a company. This example omits empty matches. Note, however, that empty matches nonetheless increment the MATCH_NUMBER:

select * from stock_price_history match_recognize( partition by company order by price_date measures match_number() as "MATCH_NUMBER" all rows per match omit empty matches pattern(OVERAVG*) define OVERAVG as price > avg(price) over (rows between unbounded preceding and unbounded following) ) order by company, price_date; +---------+------------+-------+--------------+ | COMPANY | PRICE_DATE | PRICE | MATCH_NUMBER | |---------+------------+-------+--------------| | ABCD | 2020-10-07 | 71 | 7 | | ABCD | 2020-10-08 | 80 | 7 | | ABCD | 2020-10-09 | 75 | 7 | | ABCD | 2020-10-10 | 63 | 7 | | XYZ | 2020-10-01 | 89 | 1 | | XYZ | 2020-10-04 | 63 | 4 | | XYZ | 2020-10-05 | 65 | 4 | | XYZ | 2020-10-06 | 56 | 4 | | XYZ | 2020-10-08 | 54 | 6 | +---------+------------+-------+--------------+

### Demonstrate the `WITH UNMATCHED ROWS`

Option¶

This example demonstrates the `WITH UNMATCHED ROWS option`

. Like the
Omit Empty Matches example above, this example searches for price ranges
above the average price of each company’s chart. Note that the quantifier in this query is `+`

, while the
quantifier in the previous query was `*`

:

select * from stock_price_history match_recognize( partition by company order by price_date measures match_number() as "MATCH_NUMBER", classifier() as cl all rows per match with unmatched rows pattern(OVERAVG+) define OVERAVG as price > avg(price) over (rows between unbounded preceding and unbounded following) ) order by company, price_date; +---------+------------+-------+--------------+---------+ | COMPANY | PRICE_DATE | PRICE | MATCH_NUMBER | CL | |---------+------------+-------+--------------+---------| | ABCD | 2020-10-01 | 50 | NULL | NULL | | ABCD | 2020-10-02 | 36 | NULL | NULL | | ABCD | 2020-10-03 | 39 | NULL | NULL | | ABCD | 2020-10-04 | 42 | NULL | NULL | | ABCD | 2020-10-05 | 30 | NULL | NULL | | ABCD | 2020-10-06 | 47 | NULL | NULL | | ABCD | 2020-10-07 | 71 | 1 | OVERAVG | | ABCD | 2020-10-08 | 80 | 1 | OVERAVG | | ABCD | 2020-10-09 | 75 | 1 | OVERAVG | | ABCD | 2020-10-10 | 63 | 1 | OVERAVG | | XYZ | 2020-10-01 | 89 | 1 | OVERAVG | | XYZ | 2020-10-02 | 24 | NULL | NULL | | XYZ | 2020-10-03 | 37 | NULL | NULL | | XYZ | 2020-10-04 | 63 | 2 | OVERAVG | | XYZ | 2020-10-05 | 65 | 2 | OVERAVG | | XYZ | 2020-10-06 | 56 | 2 | OVERAVG | | XYZ | 2020-10-07 | 50 | NULL | NULL | | XYZ | 2020-10-08 | 54 | 3 | OVERAVG | | XYZ | 2020-10-09 | 30 | NULL | NULL | | XYZ | 2020-10-10 | 32 | NULL | NULL | +---------+------------+-------+--------------+---------+