C#使用Linq to XML进行XPath查询
最近在用到HtmlAgliltyPack进行结点查询时,发现这里选择结点使用的是XPath。所以这里总结一下在C#中使用XPath查询XML的方法。习惯了用Linq,这里也是用的Linq to xml的。
Linq To XML的核心类是XDocument和XElement、XAttribute,需要引用using System.Xml.Linq命名空间。
这三个类简单理解为:
XDocument:打开的整个XML文档
XElement:节点元素
XAttribute:属性
下面简单介绍一下使用。
如有以下XML文件
<?xml version="1.0" encoding="utf-8"?>
<html>
<head>
<title>测试XML</title>
</head>
<body>
<div class="card-container">
<div class="item-title">
<h3 class="item-shop-name">汪家羊肉馆</h3>
</div>
<div class="item-comment">
<span class="price1">¥/人</span>
<span class="price2">¥/人</span>
<span class="price3"></span>
<span class="price4"></span>
</div>
</div>
</body>
</html>
打开XML文件
XDocument doc = XDocument.Load("demo.xml");
获取根节点(html节点 返回XElement类型)
var root = doc.Root;
获取第一个子节点(返回XElement类型)
var firstNode = root.FirstNode;
获取全部子节点(返回IEnumerable<XNode>)
var allChildNode = root.Nodes();
获取指定名称的第一个子节点(返回XElement类型)
var headNode = root.Element("head");
获取指定名称的全部子节点(返回IEnumerable<XElement>)
var allNamedNode = root.Elements("head");
获取节点指定名称的属性(<div class="card-container"> )
var attribute = root.Element("body").Element("div").Attribute("class");
获取节点的全部属性
var allAttributes = root.Attributes();
使用XPath查询(需要引用using System.Xml.XPath命名空间)
var xpathQeury = root.XPathSelectElement("body/div");
获取节点的名称和值
//获取结点的名称
var nodeName = root.Name;
//获取节点的值
var nodeValue = root.Value;
保存XML
doc.Save("demo.xml");
Linq查询(获取<span class="price2">¥12/人</span>节点下的值)
这里仅做示例,返回返回IEnumerable<XElement>类型的都可以进行Linq查询
不使用XPath
var queryResult = root.Element("body").Element("div").Elements("div").ElementAt().Elements("span").Where(x=>x.Attribute("class") != null && x.Attribute("class").Value == "price2").FirstOrDefault();
if(queryResult != null)
Console.WriteLine(queryResult.Value);
使用XPath
var xpathQueryResult = root.XPathSelectElements("body/div/div[2]/span").Where(x => x.Attribute("class") != null && x.Attribute("class").Value == "price2").FirstOrDefault();
if(xpathQueryResult != null)
Console.WriteLine(xpathQueryResult.Value);
输出结果都为下
下面开始介绍XPath
XPath 是 XML 文档中查找信息的语言,使用XPath可以对XML的元素进行查找。
这里重着介绍一下XPath语法
/ 从根节点开始选择
var xpathRoot = doc.XPathSelectElement("/html");
var xpathRoot2 = doc.XPathSelectElement("html");
运行结果
// 获取文档中所有指定的结点,不管它的位置
如要获取xml文档中所有的span节点
//以下获取的结果都是一样的
//尽管位置不一样,但查询结果是一样的
var body = root.Element("body").Element("div").Elements("div").ElementAt();
var allDivNodeInBody = body.XPathSelectElements("//div");
var allDivNodeInDoc = doc.XPathSelectElements("//div");
Console.WriteLine(allDivNodeInBody.Count());
Console.WriteLine(allDivNodeInDoc.Count());
运行结果
. 当前节点
var currentNode = root.XPathSelectElement(".");
运行结果
.. 当前节点的父节点
var parentNode = root.Element("body").XPathSelectElement("..");
运行结果
@ 选取属性
这个操作在Linq to xml里不支持,会产生一个"XPath 表达式的计算结果为意外类型 System.Xml.Linq.XAttribute。"异常
通配查询
* 全部节点
@* 全部属性(不支持)
谓词查询
选择第一个查询结果,如<span class="price1">¥62/人</span>
//span[1]
var firstSpanNode = doc.XPathSelectElement("//span[1]");
运行结果
选择最后一个查询结果,如最后一个<span class="price4">12</span>
//span[last()]
var lastDivNode = doc.XPathSelectElement("//span[last()]");
Console.WriteLine(lastDivNode.Attribute("class").Value);
运行结果
选择带有属性class的div节点
//div[@class]
var nodeWithClassAttribute = doc.XPathSelectElements("//div[@class]");
foreach (var item in nodeWithClassAttribute)
{
Console.WriteLine("====================");
Console.WriteLine(item);
}
运行结果
选择带有属性class,且值为item-title的div节点
//div[@class='item-title']
var nodeWithClassValueAttribute = doc.XPathSelectElements("//div[@class='item-title']");
foreach (var item in nodeWithClassValueAttribute)
{
Console.WriteLine(item);
}
运行结果
简单的选择计算
获取倒数第二个div节点
//div[last()-1]
var nextToLastDivNode = doc.XPathSelectElement("//div[last()-1]");
运行结果
获取值大于10的span节点
//span[text()>10]
var greaterThanTenSpan = doc.XPathSelectElements("//span[text()>10]");
运行结果:
获取值加10等于22的span节点
//span[text()+10=22]
var plusTenSpan = doc.XPathSelectElements("//span[text()+10=22]");
运行结果:
常用运行符如下
+ 加
- 减
* 乘
div 除
= 等于
!= 不等于
> 大于
< 小于
>= 大于等于
<= 小于等于
or 或
and 与
组合查询
如查询值等于12或等于2的span节点
//span[text()=2]|//span[text()=12]
var combineQuery = doc.XPathSelectElements("//span[text()=2]|//span[text()=12]");
foreach (var item in combineQuery)
{
Console.WriteLine(item.Value);
}
运行结果
参考:
https://www.w3.org/TR/2017/REC-xpath-31-20170321/
https://www.w3school.com.cn/xpath/index.asp
C#使用Linq to XML进行XPath查询的更多相关文章
- Python通过lxml库遍历xml通过xpath查询(标签,属性名称,属性值,标签对属性)
xml实例: 版本一: <?xml version="1.0" encoding="UTF-8"?><country name="c ...
- 为什么LINQ to XML的性能要优于XmlDocument?
一直很忙,压了很多贴,今天发一篇吧.后面的看心情吧. 今天群里有人问如何解析web.config方便,然后我就推荐了Linq to XML,然后就有人说“我宁可XmlDocument,再SeleteN ...
- C#基础知识---Linq操作XML文件
概述 Linq也就是Language Integrated Query的缩写,即语言集成查询,是微软在.Net 3.5中提出的一项新技术. Linq主要包含4个组件---Linq to Objects ...
- linq to xml学习
http://www.cnblogs.com/greatverve/archive/2010/07/09/linq-to-xml-add-delete-update-query.html 记录一下,别 ...
- C#3.0新增功能09 LINQ 基础01 语言集成查询
连载目录 [已更新最新开发文章,点击查看详细] 语言集成查询 (LINQ) 是一系列直接将查询功能集成到 C# 语言的技术统称. 数据查询历来都表示为简单的字符串,没有编译时类型检查或 Inte ...
- C# xml 读xml、写xml、Xpath、Xml to Linq、xml添加节点 xml修改节点
#region XDocument //创建XDocument XDocument xdoc2 = new XDocument(); XElement xel1= new XElement(" ...
- LINQ系列:LINQ to XML查询
1. 读取XML文件 XDocument和XElement类都提供了导入XML文件的Load()方法,可以读取XML文件的内容,并转换为XDocument或XElement类的实例. 示例XML文件: ...
- .NET(C#):使用XPath查询带有命名空间(有xmlns)的XML
原文http://www.cnblogs.com/mgen/archive/2011/05/24/2056025.html 众所周知,XmlDocument可以进行XPath查询,但实际上这里所说的X ...
- hibernate mapping文件中 xmlns会导致linq to xml 查询不到对应的节点
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
随机推荐
- Codeforces 1163E Magical Permutation [线性基,构造]
codeforces 思路 我顺着图论的标签点进去的,却没想到-- 可以发现排列内每一个数都是集合里的数异或出来的. 考虑答案的上界是多少.如果能用小于\(2^k\)的数构造出\([0,2^k-1]\ ...
- mysql 创建时间字段
alter table table1 add order_date datetime null; mysql> select * from table1; +----------+------- ...
- python3 报错UnicodeEncodeError
在ubuntu执行python3的时候,出现 UnicodeEncodeError: 'latin-1' codec can't encode characters in position 10-18 ...
- element ui input 输入时触发事件
<el-form-item label="客户名" :label-width="labelWidth"> <el-input v-model= ...
- SDN第六次上机作业
1.实验拓扑 实验拓扑图如下: 搭建代码如下: 创建py脚本文件,并编写代码,如下: class MyTopo(Topo): def __init__(self): # initilaize topo ...
- session机制,浏览器禁用cookie后,怎么使用session
sessionid是存储在cookie中的,解决方案如下: Session URL重写,保证在客户端禁用或不支持COOKIE时,仍然可以使用Session session机制.session机制是一种 ...
- python 玩爬虫安装了一大堆第三方库
之前就听说过爬虫,感觉很复杂的样子,但是看到python代码很简短.由于本机已经安装了python2.7 所以就拿来py 文件跑一下想看看效果. 结果各种代码错误.然后根据每个错误去下载对应的依赖项. ...
- 第2课第7节_Java面向对象编程_内部类_P【学习笔记】
摘要:韦东山android视频学习笔记 1.什么是内部类:在类的内部定义一个类,内部类可以访问类的私有属性 class Outer{ ; class Inner{ public void print ...
- PostgreSQL中的一些日志
1.Write Ahead Log(WAL日志)预写日志(WAL)是保证数据完整性的标准方法.在有关交易处理的大多数(如果不是全部)书籍中都可以找到详细说明. 简而言之,WAL的核心概念是,只有在记录 ...
- [AWS] Cloud Server
一元课程:AWS云计算——AWS操作指南系列课程 AWS 入门指南 1.1 Create one account 1.2 Create IAM Users Create 'group' firstly ...