XSLT Reference
key()
Looks up nodes using a named xsl:key index, returning all nodes whose key value matches the given value.
Syntax
key(name, value)Description
key() retrieves nodes from the source document that have been indexed by an xsl:key declaration. It is essentially a pre-built hash-map lookup: you declare the index once with xsl:key, and then query it efficiently with key().
The function is used for two main purposes:
- Cross-reference lookups — retrieving a node by an ID-like value without iterating the entire document (much faster than
//element[@id='x']in large documents). - Muenchian grouping (XSLT 1.0) — identifying the first node of each group by combining
key()withgenerate-id().
Parameters
| Parameter | Type | Required | Description |
|---|---|---|---|
name | string literal | Yes | The name of the xsl:key index to query (must match the name attribute of an xsl:key declaration). |
value | string / node-set | Yes | The key value to look up. If a node-set, the result is the union of nodes matching each node’s string value. |
Prerequisite: xsl:key declaration
<xsl:key name="key-name" match="node-pattern" use="key-expression"/>
| Attribute | Description |
|---|---|
match | Pattern identifying which nodes to index. |
use | Expression that provides the key value for each matched node. |
Return value
A node-set containing all nodes matched by match in xsl:key whose use expression equals value.
Examples
Cross-reference lookup
Input XML:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<departments>
<dept id="D1">Engineering</dept>
<dept id="D2">Marketing</dept>
</departments>
<employees>
<employee dept="D1"><name>Alice</name></employee>
<employee dept="D2"><name>Bob</name></employee>
<employee dept="D1"><name>Carol</name></employee>
</employees>
</root>
Stylesheet:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:key name="dept-by-id" match="dept" use="@id"/>
<xsl:template match="/root">
<report>
<xsl:for-each select="employees/employee">
<item>
<xsl:value-of select="name"/>
<xsl:text> — </xsl:text>
<xsl:value-of select="key('dept-by-id', @dept)"/>
</item>
</xsl:for-each>
</report>
</xsl:template>
</xsl:stylesheet>
Output:
<report>
<item>Alice — Engineering</item>
<item>Bob — Marketing</item>
<item>Carol — Engineering</item>
</report>
Muenchian grouping (XSLT 1.0)
<xsl:key name="by-category" match="product" use="category"/>
<!-- Select the first product of each distinct category -->
<xsl:for-each select="product[generate-id() = generate-id(key('by-category', category)[1])]">
<group category="{category}">
<xsl:value-of select="count(key('by-category', category))"/> items
</group>
</xsl:for-each>
Notes
key()only works within the current document by default. Usekey()combined withdocument()to look up across external documents.- The XSLT processor builds the key index once per document, making
key()significantly faster than predicate-based searches for repeated lookups. - In XSLT 2.0+,
xsl:for-each-groupreplaces Muenchian grouping with a cleaner syntax. - If
namedoes not match anyxsl:keydeclaration, the result is an error.