如何更改某些元素的XML命名空间

时间:2012-06-19 14:52:15

标签: c# .net xml xml-parsing

我有一些xml通过一些WCF消息的xmlserialization生成。 现在我想制作一个通用方法,我将提供一个xml文件名和一个像mailxml12这样的前缀。 然后在xml文件中,那些名称中没有任何名称空间前缀的元素应替换为mailxml12:

与源文件类似:

<DeliveryApptCreateRequest d2p1:ApptType="Pallet" d2p1:PickupOrDelivery="Delivery" d2p1:ShipperApptRequestID="4490660303D5" d2p1:SchedulerCRID="234234" xmlns:d2p1="http://idealliance.org/Specs/mailxml12.0a/mailxml_defs" xmlns="http://idealliance.org/Specs/mailxml12.0a/mailxml_tm">
<SubmittingParty d2p1:MailerID6="123446" d2p1:CRID="342343" d2p1:MaildatUserLicense="A123" />
<SubmittingSoftware d2p1:SoftwareName="asds" d2p1:Vendor="123" d2p1:Version="12" />
<SubmitterTrackingID>2CAD3F71B4405EB16392</SubmitterTrackingID>
<DestinationEntry>No</DestinationEntry>
<OneTimeAppt>
  <PreferredAppt>2012-06-29T09:00:00Z</PreferredAppt>
</OneTimeAppt>    
<TrailerInfo>
  <Trailer>
    <TrailerNumber>A</TrailerNumber>
    <TrailerLength>20ft</TrailerLength>
  </Trailer>
  <Carrier>
    <CarrierName>N/A</CarrierName>
    <URL>http://test.com</URL>
  </Carrier>
  <BillOfLadingNumber>N/A</BillOfLadingNumber>
</TrailerInfo>   
</DeliveryApptCreateRequest>

在所需方法之后,应将其更改为所有没有mailxml:前缀的元素名称。 与DeliveryApptCreateRequest一样应该成为mailxml:DeliveryApptCreateRequest 而像d2p1:CompanyName这样的元素应该保持不变。

我尝试使用以下代码

 private void RepalceFile(string xmlfile)
    {
        XmlDocument doc = new XmlDocument();
        doc.Load(xmlfile);
        var a = doc.CreateAttribute("xmlns:mailxml12tm");
        a.Value = "http://idealliance.org/Specs/mailxml12.0a/mailxml_tm";
        doc.DocumentElement.Attributes.Append(a);
        doc.DocumentElement.Prefix = "mailxml12tm";

        foreach (XmlNode item in doc.SelectNodes("//*"))
        {
            if (item.Prefix.Length == 0)
                item.Prefix = "mailxml12tm";
        }
        doc.Save(xmlfile);
    }

唯一的问题是根元素保持原样,而所有根据我需要改变

3 个答案:

答案 0 :(得分:2)

您可以将整个XML解析为字符串,并在适当的位置插入名称空间。但是,此解决方案可以创建许多仅在算法中使用的新字符串,这对性能不利。但是,我已经编写了一个以这种方式解析它的函数,它似乎对你发布的样本XML运行得非常快;)。如果您想使用它,我可以发布它。

另一种解决方案是将XML加载为XmlDocument并利用它是树状结构的事实。这样,您可以在适当的位置以递归方式添加适当的名称空间。 不幸的是,XmlNode.Name属性是只读的,这就是为什么你必须手动复制xml的整个结构来更改某些节点的名称的原因。 我现在没有时间编写代码,所以我只是让你写它。如果您遇到任何问题,请告诉我。

<强>更新

我已经测试了 Jeff Mercado 建议的代码和代码,并且它们似乎都正常工作,至少在您在问题中发布的示例XML中。确保您尝试解析的XML与您发布的XML相同。

为了使其工作并解决最初提出的添加命名空间问题,您可以使用代码,该代码将整个XML处理为String并手动解析:

private static String UpdateNodesWithDefaultNamespace(String xml, String defaultNamespace)
    {
        if (!String.IsNullOrEmpty(xml) && !String.IsNullOrEmpty(defaultNamespace))
        {
            int currentIndex = 0;

            while (currentIndex != -1)
            {
                //find index of tag opening character
                int tagOpenIndex = xml.IndexOf('<', currentIndex);

                //no more tag openings are found
                if (tagOpenIndex == -1)
                {
                    break;
                }

                //if it's a closing tag
                if (xml[tagOpenIndex + 1] == '/')
                {
                    currentIndex = tagOpenIndex + 1;
                }
                else
                {
                    currentIndex = tagOpenIndex;
                }

                //find corresponding tag closing character
                int tagCloseIndex = xml.IndexOf('>', tagOpenIndex);
                if (tagCloseIndex <= tagOpenIndex)
                {
                    throw new Exception("Invalid XML file.");
                }

                //look for a colon within currently processed tag
                String currentTagSubstring = xml.Substring(tagOpenIndex, tagCloseIndex - tagOpenIndex);
                int firstSpaceIndex = currentTagSubstring.IndexOf(' ');
                int nameSpaceColonIndex;
                //if space was found
                if (firstSpaceIndex != -1)
                {
                    //look for namespace colon between tag open character and the first space character
                    nameSpaceColonIndex = currentTagSubstring.IndexOf(':', 0, firstSpaceIndex);
                }
                else
                {
                    //look for namespace colon between tag open character and tag close character
                    nameSpaceColonIndex = currentTagSubstring.IndexOf(':');
                }

                //if there is no namespace
                if (nameSpaceColonIndex == -1)
                {
                    //insert namespace after tag opening characters '<' or '</'
                    xml = xml.Insert(currentIndex + 1, String.Format("{0}:", defaultNamespace));
                }

                //look for next tags after current tag closing character
                currentIndex = tagCloseIndex;
            }
        }

        return xml;
    }

您可以检查此代码以使应用程序正常工作,但是,我强烈建议您确定其他解决方案建议无效的原因。

答案 1 :(得分:1)

因为在这种情况下你定义了一个默认的命名空间,你可以删除默认的命名空间声明,并使用旧的命名空间名称为你的新前缀添加一个新的声明,从而有效地替换它。

var prefix = "mailxml";
var content = XElement.Parse(xmlStr);
var defns = content.GetDefaultNamespace();
content.Attribute("xmlns").Remove();
content.Add(new XAttribute(XNamespace.Xmlns + prefix, defns.NamespaceName));

答案 2 :(得分:1)

@ JeffMercado的解决方案对我不起作用(可能因为我没有默认命名空间)。

我最终使用了:

XNamespace ns = Constants.Namespace;
el.Name = (ns + el.Name.LocalName) as XName;

要更改我使用的整个文档的命名空间:

private void rewriteNamespace(XElement el)
{
    // Change namespace
    XNamespace ns = Constants.Namespace;
    el.Name = (ns + el.Name.LocalName) as XName;

    if (!el.HasElements)
        return;

    foreach (XElement d in el.Elements())
        rewriteNamespace(d);
}

用法:

var doc = XDocument.parse(xmlStr);
rewriteNamespace(doc.Root)

HTH