gpt4 book ai didi

c# - xpath 返回字符串而不是节点列表

转载 作者:太空宇宙 更新时间:2023-11-03 22:18:10 24 4
gpt4 key购买 nike

我正在处理一个 biztalk 项目,我需要将(过滤的)内容从 1 个 xml 复制到另一个。我必须用 xpath 来做这个,我不能使用 xsl 转换。所以我从源 xml 文件中获取内容的 xpath 是这样的:

//*[not(ancestor-or-self::IN1_Insurance)]|//IN1_Insurance[2]/descendant-or-self::*

现在这将返回一个 xmlNodelist。是否可以返回一个包含所有节点的字符串:

"<root><node>text</node></root>"

如果我将 string() 放在我的 xpath 之前,它会返回值,但我希望整个 xml 都在一个字符串中(带有节点..),所以我可以将该字符串加载到另一个 xmldocument 中。我认为这是解决我的问题的最佳方法。

我知道我可以遍历 xmlnodelist 并将节点附加到新的 xmldocument,但是在 biztalk 编排中循环有点棘手,我想避免这种情况。

我可以使用的代码是 C#。我试图将节点列表分配给 xmldocument,但这会引发转换错误(很明显..)。

我的看法是我有 2 个解决方案:

  • 在没有循环的情况下将节点列表分配给 xmldocument(我认为在 C# 中不可能)
  • 以某种方式将节点列表转换为字符串并将其加载到 xml 文档中
  • 直接在新的 xmldocument 中加载 xpath(不知道这是否可能,因为它返回一个节点列表)

谢谢你的帮助

编辑:

示例输入:

<root>
<Patient>
<PatientId></PatientId>
<name></name>
</Patient>
<insurance>
<id>1</id>
<billing></billing>
</insurance
<insurance>
<id>2</id>
<billing></billing>
</insurance>
<insurance>
<id>3</id>
<billing></billing>
</insurance>
</root>

现在我想将这个示例复制到另一个xmldocument,但是没有保险节点2和3(这是动态的,所以它可能是要删除的保险节点1和2,或者1和3 ...)

所以这必须是输出:

<root>
<Patient>
<PatientId></PatientId>
<name></name>
</Patient>
<insurance>
<id>1</id>
<billing></billing>
</insurance>
</root>

我现在正在做的是使用 xpath 来获取我想要的节点。然后我想将结果分配给新的 xmldocument,但这是不可能的,因为我得到了 castException

string xpath = "//*[not(ancestor-or-self::IN1_Insurance)]|//IN1_Insurance[2]/descendant-or-self::*";
xmlDoc = new System.Xml.XmlDocument();
xmlDoc = xpath(sourceXml, strXpath); <= cast error (cannot cast xmlnodelist to xmldocuemnt)

我知道语法有点奇怪,但它是 biztalk c# 代码..

最佳答案

最直接的解决方案确实是“遍历 ​​xmlnodelist 并将节点追加(导入)到新的 xmldocument”,但是既然你不能循环,还有什么其他基本的事情你能/不能做?

要序列化节点列表,您可以尝试使用 XmlNodeList.toString()。如果成功了,您将得到一个奇怪的野兽,因为它可能会多次复制 XML 文档的某些部分。特别是因为您直接在节点列表中明确包括祖先和后代。它不会是您可以重新解析并得到类似于您开始使用的节点列表的结果。

换句话说,最好遍历 XmlNodeList 并将节点导入新的 XmlDocument。

但即便如此,如果你想把所有这些祖先和后代节点都放在一起,我会感到非常惊讶:

//*[not(ancestor-or-self::IN1_Insurance)]|//IN1_Insurance[2]/descendant-or-self::

直接进入新的 XML 文档。如果您发布一些示例输入和所需的输出,我们可能可以帮助确定是否属于这种情况。

更新:

我明白您要做什么:复制 XML 文档,省略所有 <insurance>元素(及其后代),除了你想要的那个。

这可以在没有循环的情况下完成 if 输出与您的示例输出一样简单:只有一个 <Patient>和一个<insurance>元素及其后代位于一个顶级元素下。

类似的东西(我无法测试这个,因为我没有 biztalk 服务器):

string xpathPatient = "/*/Patient";
string xpathInsuran = "/*/insurance[id = " + insId + "]"; // insId is a parameter
xmlDoc = new System.Xml.XmlDocument();
xmlPatient = xpath(sourceXml, xpathPatient);
xmlInsuran = xpath(sourceXml, xpathInsuran);
XmlElement rootNode = xmlDoc.CreateElement("root");
xmlDoc.AppendChild(rootNode);
//**Update: use [0] to get an XmlNode from the returned XmlNodeList (presumably)
rootNode.AppendChild(xmlDoc.ImportNode(xmlPatient[0], true));
rootNode.AppendChild(xmlDoc.ImportNode(xmlInsuran[0], true));

但我承认,我很好奇您为什么不能使用 XSLT。您正在处理在 XSLT 中比在 XPath + C# XmlDocument 中更容易完成的任务。

更新:由于 xpath() 函数可能返回 XmlNodeList 而不是 XmlNode,因此我添加了 [0]上面 ImportNode() 的第一个参数。感谢@Martin Honnen 提醒我这一点。

关于c# - xpath 返回字符串而不是节点列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4246894/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com