.NET 使用 XPath 来读写 XML 文件
XPath 是 XML 路径语言(XML Path Language),用来确定XML文档中某部分位置的语言。无论是什么语言什么框架,几乎都可以使用 XPath 来高效查询 XML 文件。
本文将介绍 .NET 中的 XPath 相关类型的使用。
本文读写的 XML 文件会以 文章末尾的代码 - 假设的 XML 文件 作为示例。
关于 XPath 语法,可以阅读 XML 的 XPath 语法 了解更多。
一切从这里开始
.NET 中支持 XPath 的 XML 文档类有两种读取方法,一种是 XPathDocument
,以只读的方式读取;另一种是 XmlDocument
,不止可以读,还可以编辑。
// 得到 walterlv.xml 文档在内存中的快速只读表示形式。
var xPathDocument = new XPathDocument("walterlv.xml");
// 以可读可写的方式打开 walterlv.xml 文件。
var xmlDocument = new XmlDocument();
xmlDocument.Load("walterlv.xml");
如果要确定 XML 的文件编码,需要使用 XmlTextReader
来读 XML 文件;它的基类 XmlReader
没有提供编码信息。XmlTextReader
作为参数传入 XPathDocument
的构造函数或 XmlDocument.Load
方法中即可。
无论是 XPathDocument
还是 XmlDocument
,因为都实现了 IXPathNavigable
,所以都有 CreateNavigator();
方法,调用能得到 XPathNavigator
对象。不过前者的 CanEdit
是 false
,后者的 CanEdit
是 true
。
var navigator1 = xPathDocument.CreateNavigator();
var navigator2 = xmlDocument.CreateNavigator();
上手 XPath
路径查询
XPathNavigator
对象提供了下面两种通用的 XPath
表达式的使用检索方法。
Select
SelectSingleNode
比如希望检索本文末尾的 XML 文件中的 id
,使用 /package/metadata/id
即可检索。
当然,事实上这个 XML 文件是不能这样检索出来 id
的,因为它带有命名空间。
带有命名空间的检索需要使用到 XmlNamespaceManager
类,并写成下面这样:
var namespaceManager = new XmlNamespaceManager(new NameTable());
namespaceManager.AddNamespace("d", "http://schemas.microsoft.com/packaging/2012/06/nuspec.xsd");
navigator.Select("/d:package/d:metadata/d:id", namespaceManager);
这里其实略微奇怪,因为命名 package
、id
等都在默认的命名空间下,我们却必须显式加一个命名空间前缀。微软对此的解释是如果不指定命名空间前缀,默认都是 null
,而不是 XML 声明的那个默认命名空间。这里是原文:
XPath treats the empty prefix as the
null
namespace. In other words, only prefixes mapped to namespaces can be used in XPath queries. This means that if you want to query against a namespace in an XML document, even if it is the default namespace, you need to define a prefix for it.
路径检索的语法也有很多种,可以参考我的另一篇文章 XML 的 XPath 语法。
为了提升性能,XPathNavigator
额外提供了这些方法,用于替代 XPath
中的部分对应的语法:
SelectChildren
SelectAncestors
SelectDescendants
XPath 函数调用
Compile
和 Evaluate
提供了复杂的 XPath 函数调用。比如下面我们把几种 url 都拼接在一起得到一个新字符串。
XPathExpression query = navigator.Compile("concat(//licenseUrl/text(), //projectUrl/text(), //iconUrl/text())");
string urls = (string) navigator.Evaluate(query);
节点匹配
Matches
用来检查当前的节点是否满足某个条件。比如下面的例子便是检查当前节点的父节点是否是 group
并且其 targetFramework
属性为 .NETStandard2.0
。显然,符合这个条件的只有最后的那个 dependency
节点。
navigator.Matches("../group/@targetFramework='.NETStandard2.0'");
XPath 导航
XPathNavigator
可以在节点、属性中间移动,以便能够不止从根节点进行查询。
MoveTo
MoveToChild
MoveToFirst
MoveToFirstChild
MoveToFollowing
MoveToId
MoveToNext
MoveToParent
MoveToPrevious
MoveToRoo
MoveToAttribute
MoveToFirstAttribute
MoveToNextAttribute
MoveToNamespace
MoveToFirstNamespace
MoveToNextNamespace
在导航到需要的节点或者属性后,可以使用 navigator.OuterXml
拿到节点的所有 XML 字符串。也可以使用下面这些方法拿到节点内部的值。
ValueAsBoolean
ValueAsDateTime
ValueAsDouble
ValueAsInt
ValueAsLong
ValueAs
编辑 XML
由于我们要编辑 XML 数据,所以加载 XML 文件的方式不能是 XPathDocument
了,得是 XmlDocument
。
插入使用 Insert
相关的方法,删除使用 Delete
相关的方法。而修改数据使用 SetValue
。
保存 XML 到文件
保存 XML 使用 XmlDocument
的 Save
或者 WriteTo
方法即可。
假设的 XML 文件
<?xml version="1.0" encoding="utf-8"?>
<package xmlns="http://schemas.microsoft.com/packaging/2012/06/nuspec.xsd">
<metadata>
<id>MSTestEnhancer</id>
<version>1.6.0</version>
<authors>walterlv</authors>
<owners>walterlv</owners>
<requireLicenseAcceptance>false</requireLicenseAcceptance>
<licenseUrl>https://github.com/easiwin/MSTestEnhancer/blob/master/LICENSE</licenseUrl>
<projectUrl>https://easiwin.github.io/mstest-enhancer</projectUrl>
<iconUrl>https://easiwin.github.io/mstest-enhancer/icon.png</iconUrl>
<description>MSTestEnhancer helps you to write unit tests without naming any method. You can write method contract descriptions instead of writing confusing test method name when writing unit tests.</description>
<releaseNotes>Support passing null into WithArgument method.</releaseNotes>
<copyright>Copyright (c) 2018 dotnet职业技术学院</copyright>
<repository type="git" url="https://github.com/easiwin/MSTestEnhancer.git" />
<dependencies>
<group targetFramework=".NETFramework4.5">
<dependency id="MSTest.TestFramework" version="1.2.0" exclude="Build,Analyzers" />
<dependency id="System.ValueTuple" version="4.4.0" exclude="Build,Analyzers" />
</group>
<group targetFramework=".NETFramework4.7">
<dependency id="MSTest.TestFramework" version="1.2.0" exclude="Build,Analyzers" />
</group>
<group targetFramework=".NETStandard2.0">
<dependency id="MSTest.TestFramework" version="1.2.0" exclude="Build,Analyzers" />
</group>
</dependencies>
</metadata>
</package>
参考资料
- 使用 XPath 导航选择节点 - Microsoft Docs
- Process XML Data Using the XPath Data Model - Microsoft Docs
- XPath Queries and Namespaces - Microsoft Docs
- .NET(C#):使用XPath查询带有命名空间(有xmlns)的XML - Mgen
- .net - How to use XPath with XElement or LINQ? - Stack Overflow
.NET 使用 XPath 来读写 XML 文件的更多相关文章
- C#读写xml文件的常用方法
已知有一个XML文件(bookshop.xml)如下: <?xml version="1.0" encoding="gb2312" ?> <b ...
- PHP读写XML文件的四种方法
PHP对XML文件进行读写操作的方法一共有四种,分别是:字符串方式直接读写.DOMDocument读写. XMLWrite写和XMLReader读.SimpleXML读写,本文将依次对这四种方法进行介 ...
- Java 读写XML文件 API--org.dom4j
om4j是一个Java的XML API,类似于jdom,用来读写XML文件的.dom4j是一个十分优秀的JavaXML API,具有性能优异.功能强大和极其易使用的特点,同时它也是一个开放源代码的软件 ...
- Fixflow引擎解析(三)(模型) - 创建EMF模型来读写XML文件
Fixflow引擎解析(四)(模型) - 通过EMF扩展BPMN2.0元素 Fixflow引擎解析(三)(模型) - 创建EMF模型来读写XML文件 Fixflow引擎解析(二)(模型) - BPMN ...
- cocos2d-x 读写 xml 文件
cocos2d-x 读写 xml 文件 A product of cheungmine使用cocos2d-x开发2d游戏确实方便,但是对于一般的小游戏,经常需要的工作是UI布局设计和调整,代码改来改去 ...
- 使用XPath对象解析xml文件
使用XPath对象解析xml文件 1.DocumentBuilderFactory类 工厂API,使应用程序能从XML文档获取生成DOM对象树的解析器 其构造方法受保护,用newInstance() ...
- java通过dom读写xml文件
java通过dom读写xml文件 要读的xml文件 <?xml version="1.0" encoding="GB2312"?><学生花名册 ...
- PowerShell技巧:使用XPath语法查询XML文件
[TechTarget中国原创] XML是存储结构化数据的一个很好的途径,但是想要让数据在其中发挥作用又会有些困难.每一种语言都有其特定方式来查询XML文件中的命名空间.元素及属性.PowerShel ...
- C# 读写XML文件的方法
C# 读写XML文件的方法 一.写XML文件 XmlDocument xmlDocument = new XmlDocument();xmlDocument.AppendChild(xmlDocume ...
随机推荐
- location的部分属性
http://www.w3school.com.cn/jsref/dom_obj_location.asp location.host 可以设置或返回主机名和当前url的端口 www.w3sch ...
- 关于JNDI那点事
一.JNDI是什么? JNDI--Java 命名和目录接口(Java Naming and Directory Interface),是一组在Java应用中访问命名和目录服务的API. 二.JNDI好 ...
- Memcached incr 与 decr 命令
Memcached incr 与 decr 命令用于对已存在的 key(键) 的数字值进行自增或自减操作. incr 与 decr 命令操作的数据必须是十进制的32位无符号整数. 如果 key 不存在 ...
- zabbix自动化运维学习笔记(服务器配置)
继上次博主整理的安装后,这次是配置步骤 首先打开zabbix的安装web地址 http://xx.xx.xx.xx/zabbix/setup.php xx.xx.xx.xx是服务器的IP地址 由 ...
- WPF 回车转Tab实现跳转
1.重写窗体的KeyDown事件 protected override void OnKeyDown(KeyEventArgs e) { if (e.Key == Key.Enter) { // Mo ...
- leetcode算法总结
算法思想 二分查找 贪心思想 双指针 排序 快速选择 堆排序 桶排序 搜索 BFS DFS Backtracking 分治 动态规划 分割整数 矩阵路径 斐波那契数列 最长递增子序列 最长公共子系列 ...
- 【转载】deque双向队列
继vector和queue之后,又发现一个很好用的东西. 本篇转载自http://blog.csdn.net/morewindows/article/details/6946811 deque双向队列 ...
- Ubuntu 无法获得锁
使用ubuntu安装pip 时,出现以下错误: E: 无法获得锁 /var/cache/apt/archives/lock – open (11 资源临时不可用) E: 无法锁定下载目录 解决方法: ...
- Pandas 时间序列数据绘制X轴主要刻度和次要刻度
先上效果图吧(图中Tue表示周二): Pandas和matplotlib.dates都是使用matplotlib.units来定位刻度. matplotlib.dates可以方便的手动设置刻度,同时p ...
- Docker - 在Ubuntu 14.04 Server上的安装Docker
在 Ubuntu 14.04 Server 上安装过程是最简单的, 其满足了安装 Docker的所有要求,只需要执行如下安装脚本即可. 如果你有可能,请使用14.04版本的Ubuntu, 避免给自己挖 ...