Xpath is like regular expressions for trees - it's worth learning if you're trying to extract nodes from arbitrary locations in a document. Use xml_find_all to find all matches - if there's no match you'll get an empty result. Use xml_find_first to find a specific match - if there's no match you'll get an xml_missing node.

xml_find_all(x, xpath, ns = xml_ns(x))

xml_find_first(x, xpath, ns = xml_ns(x))

xml_find_num(x, xpath, ns = xml_ns(x))

xml_find_chr(x, xpath, ns = xml_ns(x))

xml_find_lgl(x, xpath, ns = xml_ns(x))



A document, node, or node set.


A string containing a xpath (1.0) expression.


Optionally, a named vector giving prefix-url pairs, as produced by xml_ns. If provided, all names will be explicitly qualified with the ns prefix, i.e. if the element bar is defined in namespace foo, it will be called foo:bar. (And similarly for atttributes). Default namespaces must be given an explicit name. The ns is ignored when using xml_name<- and xml_set_name.


xml_find_all always returns a nodeset: if there are no matches the nodeset will be empty. The result will always be unique; repeated nodes are automatically de-duplicated.

xml_find_first returns a node if applied to a node, and a nodeset if applied to a nodeset. The output is always the same size as the input. If there are no matches, xml_find_first will return a missing node; if there are multiple matches, it will return the first only.

xml_find_num, xml_find_chr, xml_find_lgl return numeric, character and logical results respectively.

Deprecated functions

xml_find_one() has been deprecated. Instead use xml_find_first().

See also

xml_ns_strip to remove the default namespaces


x <- read_xml("<foo><bar><baz/></bar><baz/></foo>") xml_find_all(x, ".//baz")
#> {xml_nodeset (2)} #> [1] <baz/> #> [2] <baz/>
xml_path(xml_find_all(x, ".//baz"))
#> [1] "/foo/bar/baz" "/foo/baz"
# Note the difference between .// and // # // finds anywhere in the document (ignoring the current node) # .// finds anywhere beneath the current node (bar <- xml_find_all(x, ".//bar"))
#> {xml_nodeset (1)} #> [1] <bar>\n <baz/>\n</bar>
xml_find_all(bar, ".//baz")
#> {xml_nodeset (1)} #> [1] <baz/>
xml_find_all(bar, "//baz")
#> {xml_nodeset (2)} #> [1] <baz/> #> [2] <baz/>
# Find all vs find one ----------------------------------------------------- x <- read_xml("<body> <p>Some <b>text</b>.</p> <p>Some <b>other</b> <b>text</b>.</p> <p>No bold here!</p> </body>") para <- xml_find_all(x, ".//p") # If you apply xml_find_all to a nodeset, it finds all matches, # de-duplicates them, and returns as a single list. This means you # never know how many results you'll get xml_find_all(para, ".//b")
#> {xml_nodeset (3)} #> [1] <b>text</b> #> [2] <b>other</b> #> [3] <b>text</b>
# xml_find_first only returns the first match per input node. If there are 0 # matches it will return a missing node xml_find_first(para, ".//b")
#> {xml_nodeset (3)} #> [1] <b>text</b> #> [2] <b>other</b> #> [3] <NA>
xml_text(xml_find_first(para, ".//b"))
#> [1] "text" "other" NA
# Namespaces --------------------------------------------------------------- # If the document uses namespaces, you'll need use xml_ns to form # a unique mapping between full namespace url and a short prefix x <- read_xml(' <root xmlns:f = "http://foo.com" xmlns:g = "http://bar.com"> <f:doc><g:baz /></f:doc> <f:doc><g:baz /></f:doc> </root> ') xml_find_all(x, ".//f:doc")
#> {xml_nodeset (2)} #> [1] <f:doc>\n <g:baz/>\n</f:doc> #> [2] <f:doc>\n <g:baz/>\n</f:doc>
xml_find_all(x, ".//f:doc", xml_ns(x))
#> {xml_nodeset (2)} #> [1] <f:doc>\n <g:baz/>\n</f:doc> #> [2] <f:doc>\n <g:baz/>\n</f:doc>