PHP Conference Japan 2024

DOMXPath::evaluate

(PHP 5 >= 5.1.0, PHP 7, PHP 8)

DOMXPath::evaluate 评估给定的 XPath 表达式,并在可能的情况下返回类型化结果

描述

public DOMXPath::evaluate(string $expression, ?DOMNode $contextNode = null, bool $registerNodeNS = true): mixed

执行给定的 XPath expression,并在可能的情况下返回类型化结果。

参数

expression

要执行的 XPath 表达式。

contextNode

可选的 contextNode 可以指定用于执行相对 XPath 查询。默认情况下,查询相对于根元素。

registerNodeNS

是否自动将上下文节点的作用域内的命名空间前缀注册到 DOMXPath 对象。这可以用于避免需要手动为每个作用域内的命名空间调用 DOMXPath::registerNamespace()。当存在命名空间前缀冲突时,只会注册最近的子代命名空间前缀。

返回值

如果可能,返回类型化结果,否则返回一个 DOMNodeList,其中包含与给定 XPath expression 匹配的所有节点。

如果 expression 格式错误或 contextNode 无效,DOMXPath::evaluate() 返回 false

示例

示例 #1 获取所有英文书籍的数量

<?php

$doc
= new DOMDocument;

$doc->load('book.xml');

$xpath = new DOMXPath($doc);

$tbody = $doc->getElementsByTagName('tbody')->item(0);

// 我们的查询相对于 tbody 节点
$query = 'count(row/entry[. = "en"])';

$entries = $xpath->evaluate($query, $tbody);
echo
"共有 $entries 本英文书籍\n";

?>

以上示例将输出

There are 2 english books

参见

添加备注

用户贡献的笔记 5 条笔记

daniel at danielnorton dot com
13 年前
请注意,此方法不提供任何方法来区分成功返回 FALSE 的结果和错误。

例如,这将成功并返回 FALSE

<?php $xpath->evaluate("1 = 0"); ?>

当您知道期望布尔值时,一种解决方法是用 string() 包装结果。例如

<?php $xpath->evaluate("string(1 = 0)"); ?>

这将在成功时返回字符串“false”,或在错误时返回布尔值 FALSE。
Damien Bezborodov
13 年前
如果您的表达式返回节点集,您将获得 DOMNodeList 而不是类型化结果。相反,尝试将您的表达式从 "//node[1]" 修改为 "string(//node[1])"。
yuriucsal at NOSPAM dot yahoo dot com dot br
19 年前
此类可以在未验证时替换 evaluate 方法。为 Yuri Bastos 和 Jo�o Gilberto Magalh�es 制作。



<?php

class XPtahQuery
{
// 函数返回一个来自相对XPath的DOMNodeList
public static function selectNodes($pNode, $xPath)
{

$pos = strpos(self::getFullXpath($pNode),"/",1);
$xPathQuery = substr(self::getFullXpath($pNode),$pos);//粘贴/#document[1]/
$xPathQueryFull = $xPathQuery. $xPath;
$domXPath = new DOMXPath($pNode->ownerDocument);
$rNodeList = $domXPath->query($xPathQueryFull);

return
$rNodeList;
}
// 函数返回一个来自其他DOMNode的XPath的DOMNode
public static function selectSingleNode($pNode, $xPath)
{

$pos = strpos(self::getFullXpath($pNode),"/",1);
$xPathQuery = substr(self::getFullXpath($pNode),$pos);//粘贴/#document[1]/
$xPathQueryFull = $xPathQuery. $xPath;
$domXPath = new DOMXPath($pNode->ownerDocument);
$rNode = $domXPath->query($xPathQueryFull)->item(0);

return
$rNode;
}
//selectSingleNode的实用程序函数
private function getNodePos($pNode, $nodeName)
{
if(
$pNode == null)
{
return
0;
}
else
{
$var = 0;
if (
$pNode->previousSibling != null)
{
if (
$pNode->previousSibling->nodeName == $nodeName)
{
$var = 1;
}
}
return
self::getNodePos($pNode->previousSibling, $nodeName) + $var;
}
}
//selectSingleNode的实用程序函数
private function getFullXpath($pNode)
{
if(
$pNode == null)
{
return
"";
}
else
{

return
self::getFullXpath($pNode->parentNode) . "/" . $pNode->nodeName . "[" .strval(self::getNodePos($pNode, $pNode->nodeName)+1) . "]";//+1获取真实的XPath索引

}
}
}
?>
aazaharov81 at gmail dot com
9年前
要通过HTML类查询DOMNodes,请使用此代码片段
<?php

// CssClassXPathSelector
function ccxs($class) {
return
'[contains(concat(" ", normalize-space(@class), " "), " ' . $class . ' ")]';
}

// 然后只需
$domitems = $this->xpath("//*[@id='searchResultsRows']//a" . ccxs('listing_row'));
?>
danny at webdevelopers dot eu
4年前
区分返回的FALSE值与语法错误FALSE的唯一方法是重新运行用string()函数包装的XPath表达式。如果必须返回空字符串。如果它再次返回FALSE,则表示错误。

<?php

$ret
=$this->xp->evaluate($eval, $context);

// 错误检测:DOMXPath::evaluate() 在错误时返回 FALSE
// DOMXPath::evaluate("boolean(/nothing)") 也是如此
// @workaround webdevelopers.eu
if ($ret === false && $this->xp->evaluate("string($eval)", $context) === false) {
throw new
Exception("无效的XPath表达式 ".json_encode($eval), 3491);
}
?>
To Top