XPATH查询结果不一致

时间:2012-09-10 19:24:56

标签: c# xpath xml-namespaces

我正在运行转换过程来更新我们的一些xml文件。这是一个示例文件:

<?xml version="1.0" encoding="utf-8"?>
<Jobs xmlns="urn:mynamespace" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Job name="Job1">
    <Category>Maintenance</Category>
    <Description>Purge records</Description>
    <Steps>
      <TSql name="Start Job" database="IADS">
        <CommandText>Exec StoredProcedureName</CommandText>
        <OnSuccess action="GotoNextStep" />
        <Retries>0</Retries>
        <OnFailure action="QuitFailure" />
        <OutputFile />
      </TSql>
      <TSql name="Start LoadRequestManagementReportTables" database="msdb">
        <CommandText>exec sp_start_job @job_name = 'Load Request Management Report Tables'</CommandText>
        <OnSuccess action="QuitSuccess" />
        <Retries waitInMinutes="0">0</Retries>
        <OnFailure action="QuitFailure" />
        <OutputFile />
      </TSql>
    </Steps>
    <Schedules>
      <Schedule>
        <Weekly name="Every Sunday at 7:00 AM" enabled="false">
          <BeginDate>2008-11-01</BeginDate>
          <RunTimes>
            <Once>07:00</Once>
          </RunTimes>
          <DaysOfWeek>Sunday</DaysOfWeek>
        </Weekly>
      </Schedule>
      <Schedule>
        <OneTime name="OneTime" enabled="false" rundate="2011-01-12T03:00:00" />
      </Schedule>
    </Schedules>
    <Notifications>
      <EventLog on="Failure" />
    </Notifications>
  </Job>
</Jobs>

现在,我正在尝试重命名我处理的每个文件中的第一步,以将其从TSql步骤转换为CmdExec步骤。因此,我编写了自己的“RenameNode”方法来执行此转换,并将所有属性和节点复制到新命名的节点。

这是RenameNode方法:

private static void RenameNode(XmlNode node, string namespaceURI, string newName)
{
    if (node.NodeType != XmlNodeType.Element)
        return;

    XmlElement oldElement = (XmlElement)node;
    XmlElement newElement = node.OwnerDocument.CreateElement(newName, namespaceURI);

    while (oldElement.HasAttributes)
        newElement.SetAttributeNode(oldElement.RemoveAttributeNode(oldElement.Attributes[0]));

    while (oldElement.HasChildNodes)
        newElement.AppendChild(oldElement.FirstChild);

    if (oldElement.ParentNode != null)
        oldElement.ParentNode.ReplaceChild(newElement, oldElement);
}

我遇到的问题是我运行的第二个查询没有返回结果。

这是我运行的查询#1:

XmlNodeList stepNodes = xSchedule.SelectNodes("/mns:Jobs/mns:Job/mns:Steps/mns:TSql", nsm);

这很好用,在我的stepNodes变量中返回2个节点。我为stepNodes [0]处理“RenameNode”方法。奇妙。

我要做的下一件事是删除“RenameNode”方法生成的命名空间属性(我猜你不会喜欢那个......但这不是问题)。所以,为了做到这一点,我尝试运行一个非常相似的XPATH查询,看起来像这样,但它不会返回任何记录:

stepNodes = xSchedule.SelectNodes("/mns:Jobs/mns:Job/mns:Steps/mns:CmdExec", nsm);

我已经尝试重置命名空间管理器,我尝试保存xml文件(xSchedule.Save()),尝试运行带有和不带名称空间前缀/参数等的查询。它永远不会返回任何节点。即使在对节点进行重命名之后,在保存文档之后我也可以看到它成功重命名了该节点(从而将xmlns属性添加到该新命名节点中的EVERYTHING)。我甚至验证了返回结果的原始查询现在只返回1个节点。

// Let's assume that the very first node is the node that we want to change
        XmlNodeList stepNodes = xSchedule.SelectNodes("/mvst:Jobs/mvst:Job/mvst:Steps/mvst:TSql", nsm);
    if (stepNodes.Count >= 1)
    {
        RenameNode(stepNodes[0], String.Empty, "CmdExec");

        // After renaming the node, let's remove the "database" attribute if it exists
        //XmlElement e = (XmlElement)stepNodes[0];
        //e.RemoveAttribute("database");
        xSchedule.Save(scheduleXmlFile + ".bak");
    }

这个代码我哪里错了?

1 个答案:

答案 0 :(得分:0)

只需要将命名空间信息传递给RenameNode方法。

RenameNode(stepNodes[0], "urn:mynamespace", "CmdExec");