如何避免添加XML元素中的xmlns =“”?

时间:2013-11-19 14:51:55

标签: c# .net xml xml-namespaces xmldocument

我有287个SSRS报告(XML文件),我需要添加一个参数 他们都是这样开始的:

<?xml version="1.0" encoding="utf-8"?>
<Report xmlns="http://schemas.microsoft.com/sqlserver/reporting/2005/01/reportdefinition" xmlns:rd="http://schemas.microsoft.com/SQLServer/reporting/reportdesigner">
<DataSources>
<DataSource Name="COR_Basic">
<rd:DataSourceID>addde073-f37c-4b59-ae3a-25231ffc0ec6</rd:DataSourceID>
<DataSourceReference>COR_Basic</DataSourceReference>
</DataSource>
</DataSources>
<InteractiveHeight>29.7cm</InteractiveHeight>
<ReportParameters>
<ReportParameter Name="in_report_name">
<DataType>String</DataType>
<DefaultValue>
<Values>
<Value>=Globals!ReportName</Value>
</Values>
</DefaultValue>
<Prompt>Report Name</Prompt>
<Hidden>true</Hidden>
</ReportParameter>
<ReportParameter Name="in_mandant">
<DataType>String</DataType>
<DefaultValue>
<Values>
<Value>0</Value>
</Values>
</DefaultValue>
<Prompt>in_mandant</Prompt>
<Hidden>true</Hidden>
</ReportParameter>

现在我想自动将“proc”参数添加到所有报告中。
这意味着我需要插入这一部分XML

<ReportParameter Name="proc">
<DataType>String</DataType>
<DefaultValue>
<Values>
<Value>MyTestValue</Value>
</Values>
</DefaultValue>
<Prompt>MyPromptText</Prompt>
<Hidden>true</Hidden>
</ReportParameter>

之后

<ReportParameter Name="in_mandant">
...
</ReportParameter>

这是我到目前为止的代码:

public static System.Xml.XmlDocument File2XmlDocument(string strFileName)
{
    // http://blogs.msdn.com/b/tolong/archive/2007/11/15/read-write-xml-in-memory-stream.aspx
    System.Xml.XmlDocument doc = new System.Xml.XmlDocument();
    // doc.Load(memorystream);
    // doc.Load(FILE_NAME);

    using (System.Xml.XmlTextReader xtrReader = new System.Xml.XmlTextReader(strFileName))
    {

        doc.Load(xtrReader);
        xtrReader.Close();
    } // End Using xtrReader

    return doc;
} // End Function File2XmlDocument




        public static System.Xml.XmlNamespaceManager GetReportNamespaceManager(System.Xml.XmlDocument doc)
        {
            System.Xml.XmlNamespaceManager nsmgr = new System.Xml.XmlNamespaceManager(doc.NameTable);
            nsmgr.AddNamespace("dft", "http://schemas.microsoft.com/sqlserver/reporting/2005/01/reportdefinition");

            return nsmgr;
        } // End Function GetReportNamespaceManager



 public static void AddProc(string strFilename)
        {
            System.Xml.XmlDocument doc = File2XmlDocument(strFilename);
            System.Xml.XmlElement root = doc.DocumentElement;
            System.Xml.XmlNamespaceManager nsmgr = GetReportNamespaceManager(doc);

            System.Xml.XmlNode xnMandant = root.SelectSingleNode("/dft:Report/dft:ReportParameters/dft:ReportParameter[@Name=\"in_mandant\"]", nsmgr);

            string strReportName = System.IO.Path.GetFileNameWithoutExtension(strFilename);

            if (xnMandant != null)
            {
                LogMessage("{0}\t{1}", strReportName, xnMandant.FirstChild.Value);

                string frag = @"<ReportParameter Name=""proc"">
<DataType>String</DataType>
<DefaultValue>
<Values>
<Value>MyTestValue</Value>
</Values>
</DefaultValue>
<Prompt>MyPromptText</Prompt>
<Hidden>true</Hidden>
</ReportParameter>";

                System.Xml.XmlDocumentFragment xmlDocFrag = doc.CreateDocumentFragment();
                xmlDocFrag.InnerXml = frag;

                System.Xml.XmlNode xn = xnMandant.ParentNode.InsertAfter(xmlDocFrag, xnMandant);

                Console.WriteLine(xn.OuterXml);

            }
            else
                LogMessage("{0}\tKein Parameter in_mandant", strReportName);


            SaveDocument(doc, strFilename);
        } // End Sub PrintStichtag

这会在正确的位置插入XML片段,但xmlns=""之后我有一个"proc"

<ReportParameter Name="proc" xmlns="">
<DataType>String</DataType>
<DefaultValue>
<Values>
<Value>MyTestValue</Value>
</Values>
</DefaultValue>
<Prompt>MyPromptText</Prompt>
<Hidden>true</Hidden>
</ReportParameter>

如何避免空的xmlns属性?
(没有将xml文件作为文本加载并执行string.replace)。

3 个答案:

答案 0 :(得分:2)

  

如何避免空的xmlns属性?

在右侧命名空间中创建元素。由于namespace defaulting所有,您的元素应位于http://schemas.microsoft.com/sqlserver/reporting/2005/01/reportdefinition命名空间中。

您可以尝试插入片段,然后设置其InnerXml值 - 这可能会适当地应用命名空间默认值。

然而,就个人而言,我会使用LINQ to XML,这使得命名空间处理变得更加简单 - 然后以编程方式构造元素,而不是从文本中解析它们。实际上,无论如何我都会以编程方式构造元素,而不是设置InnerXml - 这样可以更容易地精确控制命名空间等内容,因为您可以在构造元素时指定它们。

(我还强烈建议使用using指令,以避免代码在任何地方都被称为完全限定名称。)

答案 1 :(得分:0)

似乎这根本不可能(“谢谢”只读属性) 围绕它的唯一方法是在文档的OuterXml上进行替换,然后重新加载文档。

    public static void SaveDocument(System.Xml.XmlDocument doc, string strFilename, bool bDoReplace)
    {
        string strSavePath = GetSavePath();
        strSavePath = System.IO.Path.Combine(strSavePath, System.IO.Path.GetFileName(strFilename));

        if (bDoReplace)
        {
            doc.LoadXml(doc.OuterXml.Replace("xmlns=\"\"", ""));
            // doc.LoadXml(doc.OuterXml.Replace(strTextToReplace, strReplacementText));
        }

        using (System.Xml.XmlTextWriter xtw = new System.Xml.XmlTextWriter(strSavePath, System.Text.Encoding.UTF8))
        {
            xtw.Formatting = System.Xml.Formatting.Indented; // if you want it indented
            xtw.Indentation = 4;
            xtw.IndentChar = ' ';

            doc.Save(xtw);
        } // End Using xtw

    } // End Sub SaveDocument

答案 2 :(得分:0)

添加时使用此命名空间进行以下设置。

     XmlSerializerNamespaces namespaces = new XmlSerializerNamespaces();
     namespaces.Add(string.Empty, string.Empty);