FIRST_VALUE [Analytic]
Lets you select the first value of a table or partition (determined by the window-order-clause) without having to use a self join. This function is useful when you want to use the first value as a baseline in calculations.
Use FIRST_VALUE()
with the window-order-clause to produce deterministic results. If no window is specified for the current row, the default window is UNBOUNDED PRECEDING AND CURRENT ROW
.
Behavior Type
Syntax
FIRST_VALUE ( expression [ IGNORE NULLS ] ) OVER ( [ window-partition-clause ] [ window-order-clause ] [ window-frame-clause ] )
Parameters
expression |
Expression to evaluate—or example, a constant, column, nonanalytic function, function expression, or expressions involving any of these. |
IGNORE NULLS
|
Specifies to return the first non-null value in the set, or |
OVER()
|
Examples
The following query asks for the first value in the partitioned day of week, and illustrates the potential nondeterministic nature of FIRST_VALUE()
:
=> SELECT calendar_year, date_key, day_of_week, full_date_description, FIRST_VALUE(full_date_description) OVER(PARTITION BY calendar_month_number_in_year ORDER BY day_of_week) AS "first_value" FROM date_dimension WHERE calendar_year=2003 AND calendar_month_number_in_year=1;
The first value returned is January 31, 2003; however, the next time the same query is run, the first value might be January 24 or January 3, or the 10th or 17th. This is because the analytic ORDER BY
column day_of_week
returns rows that contain ties (multiple Fridays). These repeated values make the ORDER BY
evaluation result nondeterministic, because rows that contain ties can be ordered in any way, and any one of those rows qualifies as being the first value of day_of_week
.
calendar_year | date_key | day_of_week | full_date_description | first_value --------------+----------+-------------+-----------------------+------------------ 2003 | 31 | Friday | January 31, 2003 | January 31, 2003 2003 | 24 | Friday | January 24, 2003 | January 31, 2003 2003 | 3 | Friday | January 3, 2003 | January 31, 2003 2003 | 10 | Friday | January 10, 2003 | January 31, 2003 2003 | 17 | Friday | January 17, 2003 | January 31, 2003 2003 | 6 | Monday | January 6, 2003 | January 31, 2003 2003 | 27 | Monday | January 27, 2003 | January 31, 2003 2003 | 13 | Monday | January 13, 2003 | January 31, 2003 2003 | 20 | Monday | January 20, 2003 | January 31, 2003 2003 | 11 | Saturday | January 11, 2003 | January 31, 2003 2003 | 18 | Saturday | January 18, 2003 | January 31, 2003 2003 | 25 | Saturday | January 25, 2003 | January 31, 2003 2003 | 4 | Saturday | January 4, 2003 | January 31, 2003 2003 | 12 | Sunday | January 12, 2003 | January 31, 2003 2003 | 26 | Sunday | January 26, 2003 | January 31, 2003 2003 | 5 | Sunday | January 5, 2003 | January 31, 2003 2003 | 19 | Sunday | January 19, 2003 | January 31, 2003 2003 | 23 | Thursday | January 23, 2003 | January 31, 2003 2003 | 2 | Thursday | January 2, 2003 | January 31, 2003 2003 | 9 | Thursday | January 9, 2003 | January 31, 2003 2003 | 16 | Thursday | January 16, 2003 | January 31, 2003 2003 | 30 | Thursday | January 30, 2003 | January 31, 2003 2003 | 21 | Tuesday | January 21, 2003 | January 31, 2003 2003 | 14 | Tuesday | January 14, 2003 | January 31, 2003 2003 | 7 | Tuesday | January 7, 2003 | January 31, 2003 2003 | 28 | Tuesday | January 28, 2003 | January 31, 2003 2003 | 22 | Wednesday | January 22, 2003 | January 31, 2003 2003 | 29 | Wednesday | January 29, 2003 | January 31, 2003 2003 | 15 | Wednesday | January 15, 2003 | January 31, 2003 2003 | 1 | Wednesday | January 1, 2003 | January 31, 2003 2003 | 8 | Wednesday | January 8, 2003 | January 31, 2003 (31 rows)
The day_of_week
results are returned in alphabetical order because of lexical rules. The fact that each day does not appear ordered by the 7-day week cycle (for example, starting with Sunday followed by Monday, Tuesday, and so on) has no affect on results.
To return deterministic results, modify the query so that it performs its analytic ORDER BY
operations on a unique field, such as date_key
:
=> SELECT calendar_year, date_key, day_of_week, full_date_description,
FIRST_VALUE(full_date_description) OVER
(PARTITION BY calendar_month_number_in_year ORDER BY date_key
) AS "first_value"
FROM date_dimension WHERE calendar_year=2003;
FIRST_VALUE()
returns a first value of January 1 for the January partition and the first value of February 1 for the February partition. Also, the full_date_description
column contains no ties:
calendar_year | date_key | day_of_week | full_date_description | first_value ---------------+----------+-------------+-----------------------+------------ 2003 | 1 | Wednesday | January 1, 2003 | January 1, 2003 2003 | 2 | Thursday | January 2, 2003 | January 1, 2003 2003 | 3 | Friday | January 3, 2003 | January 1, 2003 2003 | 4 | Saturday | January 4, 2003 | January 1, 2003 2003 | 5 | Sunday | January 5, 2003 | January 1, 2003 2003 | 6 | Monday | January 6, 2003 | January 1, 2003 2003 | 7 | Tuesday | January 7, 2003 | January 1, 2003 2003 | 8 | Wednesday | January 8, 2003 | January 1, 2003 2003 | 9 | Thursday | January 9, 2003 | January 1, 2003 2003 | 10 | Friday | January 10, 2003 | January 1, 2003 2003 | 11 | Saturday | January 11, 2003 | January 1, 2003 2003 | 12 | Sunday | January 12, 2003 | January 1, 2003 2003 | 13 | Monday | January 13, 2003 | January 1, 2003 2003 | 14 | Tuesday | January 14, 2003 | January 1, 2003 2003 | 15 | Wednesday | January 15, 2003 | January 1, 2003 2003 | 16 | Thursday | January 16, 2003 | January 1, 2003 2003 | 17 | Friday | January 17, 2003 | January 1, 2003 2003 | 18 | Saturday | January 18, 2003 | January 1, 2003 2003 | 19 | Sunday | January 19, 2003 | January 1, 2003 2003 | 20 | Monday | January 20, 2003 | January 1, 2003 2003 | 21 | Tuesday | January 21, 2003 | January 1, 2003 2003 | 22 | Wednesday | January 22, 2003 | January 1, 2003 2003 | 23 | Thursday | January 23, 2003 | January 1, 2003 2003 | 24 | Friday | January 24, 2003 | January 1, 2003 2003 | 25 | Saturday | January 25, 2003 | January 1, 2003 2003 | 26 | Sunday | January 26, 2003 | January 1, 2003 2003 | 27 | Monday | January 27, 2003 | January 1, 2003 2003 | 28 | Tuesday | January 28, 2003 | January 1, 2003 2003 | 29 | Wednesday | January 29, 2003 | January 1, 2003 2003 | 30 | Thursday | January 30, 2003 | January 1, 2003 2003 | 31 | Friday | January 31, 2003 | January 1, 2003 2003 | 32 | Saturday | February 1, 2003 | February 1, 2003 2003 | 33 | Sunday | February 2, 2003 | February 1,2003 ... (365 rows)