XPath的not()函数用于对给定的表达式进行逻辑取反操作,返回与表达式结果相反的逻辑值,是编写复杂筛选条件时常用的函数之一。它的核心作用是将原本为真的条件转为假,原本为假的条件转为真,从而实现对节点的反向筛选。

not()函数的基本语法
not()函数的语法格式非常简单,接收一个表达式作为参数,语法如下:
not(表达式)
当传入的表达式返回结果为真时,not()函数返回假;当表达式返回结果为假时,not()函数返回真。这里的真对应XPath中的布尔值true,假对应布尔值false。
否定不同场景表达式的方法
1. 否定节点存在性判断
如果要筛选不存在某个子节点的父节点,可以将节点存在判断表达式传入not()函数。例如筛选所有不包含<price>子节点的<book>节点:
//book[not(price)]
这里price表达式判断当前<book>节点下是否存在<price>子节点,存在则返回真,经过not()取反后,就只会匹配不存在<price>子节点的<book>节点。
2. 否定属性匹配条件
如果要筛选属性不满足某个条件的节点,可以将属性判断表达式传入not()函数。例如筛选所有class属性不等于highlight的<div>节点:
//div[not(@class='highlight')]
这里@class='highlight'判断当前<div>的class属性是否为highlight,经过not()取反后,就会匹配class属性不是highlight,或者没有class属性的<div>节点。
3. 否定位置筛选条件
如果要筛选位置不满足某个条件的节点,可以将位置表达式传入not()函数。例如筛选所有不是第一个子节点的<li>元素:
//li[not(position()=1)]
这里position()=1判断当前<li>是否为父节点的第一个子节点,经过not()取反后,就会匹配所有位置不是第一的<li>节点。
4. 否定多个条件的组合
如果要否定多个条件的组合结果,可以将组合条件用括号包裹后传入not()函数。例如筛选所有既不是class为active,也不是disabled状态的<button>节点:
//button[not(@class='active' or @disabled)]
这里@class='active' or @disabled判断按钮是否满足其中一个条件,经过not()取反后,就会匹配两个都不满足的<button>节点。
常见使用误区与注意事项
- 不要混淆not()函数和!=运算符:!=只能用于单个值的比较,而not()可以对任意表达式取反,例如
//div[not(@class)]是筛选没有class属性的div,而//div[@class!='test']会匹配class属性不等于test的div,但不会匹配没有class属性的div。 - not()函数的参数必须是完整的表达式:不能直接写
//book[not price],必须写成//book[not(price)],否则语法会报错。 - 注意表达式的优先级:如果组合多个条件,建议用括号明确优先级,避免逻辑判断出错,例如
//div[not(@class='a' and @id='b')]和//div[not(@class='a') and not(@id='b')]的逻辑是不同的,前者是取反两个条件的与结果,后者是两个取反条件的与结果。
实际使用示例
假设我们有如下的XML文档结构:
<library>
<book id="1">
<title>XPath教程</title>
<price>59.9</price>
</book>
<book id="2">
<title>XML基础</title>
</book>
<book id="3">
<title>HTML进阶</title>
<price>49.9</price>
</book>
</library>
如果要获取所有没有<price>子节点的book的title文本,可以使用如下XPath表达式:
//book[not(price)]/title/text()
运行该表达式后,会返回"XML基础"这个文本内容,符合我们的筛选预期。