指示XmlWriterSettings使用自闭合标签

11

我正在使用XmlWriterSettings将Xml写入文件。 我有仅具有属性而没有子元素的元素。 我希望它们输出为:

<element a="1" /> 
代替
<element a="1"></element>

我能使用XmlWriterSettings完成这个任务吗?

编辑:

以下是代码:

private void Mission_Save(string fileName)
    {
        StreamWriter streamWriter = new StreamWriter(fileName, false);
        streamWriter.Write(Mission_ToXml());
        streamWriter.Close();
        streamWriter.Dispose();

        _MissionFilePath = fileName;
    }

private string Mission_ToXml()
    {
        XmlDocument xDoc;
        XmlElement root;
        XmlAttribute xAtt;

        xDoc = new XmlDocument();

        foreach (string item in _MissionCommentsBefore)
            xDoc.AppendChild(xDoc.CreateComment(item));

        root = xDoc.CreateElement("mission_data");
        xAtt = xDoc.CreateAttribute("version");
        xAtt.Value = "1.61";
        root.Attributes.Append(xAtt); 
        xDoc.AppendChild(root);

        //Out the xml's!
        foreach (TreeNode node in _FM_tve_Mission.Nodes)
            Mission_ToXml_private_RecursivelyOut(root, xDoc, node);

        foreach (string item in _MissionCommentsAfter)
            xDoc.AppendChild(xDoc.CreateComment(item));


        //Make this look good
        StringBuilder sb = new StringBuilder();
        XmlWriterSettings settings = new XmlWriterSettings();

        settings.Indent = true;
        settings.IndentChars = "  ";
        settings.NewLineChars = "\r\n";
        settings.NewLineHandling = NewLineHandling.Replace;
        settings.OmitXmlDeclaration = true;
        using (XmlWriter writer = XmlWriter.Create(sb, settings))
        {
            xDoc.Save(writer);
        }

        return sb.ToString();
    }

private void Mission_ToXml_private_RecursivelyOut(XmlNode root, XmlDocument xDoc, TreeNode tNode)
    {
        root.AppendChild(((MissionNode)tNode.Tag).ToXml(xDoc));
        foreach (TreeNode node in tNode.Nodes)
            Mission_ToXml_private_RecursivelyOut(root, xDoc, node);
    }

这里的 _FM_tve_Mission 是一个TreeView控件,它拥有节点,每个节点都有一个MissionNode类的标签,该类具有ToXml方法,返回包含此MissionNode转换为XML的XmlNode。


你的代码是什么?没有特定设置的XmlWriter以你想要的方式进行编写。 - Paolo Falabella
3个回答

10

你不需要任何特殊的设置:

XmlWriter output = XmlWriter.Create(filepath);
 output.writeStartElement("element");
 output.writeAttributeString("a", "1");
 output.writeEndElement();

这将为您输出<element a="1" />(我在编写XML的应用程序中进行了测试)

基本上,如果您在写结束元素之前没有添加任何数据,它将自动为您关闭。

我还有以下XmlWriterSettings,如果默认情况下不起作用,可以尝试使用其中之一:

XmlWriterSettings wSettings = new XmlWriterSettings();
wSettings.Indent = true;
wSettings.ConformanceLevel = ConformanceLevel.Fragment;
wSettings.OmitXmlDeclaration = true;
XmlWriter output = XmlWriter.Create(filePathXml, wSettings);

XmlWriter只有一个静态的Create方法。你不能调用XmlWriter.Write()因为它不存在。你必须逐个写入每个节点的名称和属性。从你给出的解释来看,这正是你想要的。你不能神奇地将数据传递给xmlwriter并期望输出。你必须告诉它要输出哪些元素和属性。 - jzworkman
啊,抱歉我错了,它叫做Save而不是Write...我的意思是使用类型为XmlDocument的对象的方法Save(传递XmlWriter对象),我已经将代码添加到OP中。 - Istrebitel
3
我找到了问题所在。我将XmlNode的InnerText赋值为""。这个动作显然会使它“展开”成为<></>形式。如果不进行赋值,它仍然保持为""但是没有子元素就不会展开。问题得以解决。 - Istrebitel
你应该引用 XmlWriterSettings.WriteEndDocumentOnClose 属性 - Cole Tobin
@ColeJohnson 是的,现在可以在 .Net 4.5 中使用该设置。当我回答这个问题时,该属性不存在(.Net 4.0)。但是现在它被添加到框架中是一个很好的补充。 - jzworkman
显示剩余5条评论

3

从外部文件处理XML时,我编写了以下类以去除非空闭合元素。现在我的XML具有自闭合标签。

using System.Linq;
using System.Xml.Linq;

namespace XmlBeautifier
{
    public class XmlBeautifier
    {
        public static string BeautifyXml(string outerXml)
        {
            var _elementOriginalXml = XElement.Parse(outerXml);
            var _beautifiedXml = CloneElement(_elementOriginalXml);
            return _beautifiedXml.ToString();
        }

        public static XElement CloneElement(XElement element)
        {
            // http://blogs.msdn.com/b/ericwhite/archive/2009/07/08/empty-elements-and-self-closing-tags.aspx
            return new XElement(element.Name,
                element.Attributes(),
                element.Nodes().Select(n =>
                {
                    XElement e = n as XElement;
                    if (e != null)
                        return CloneElement(e);
                    return n;
                })
            );
        }

    }
}

XmlNode.Clone(deep_or_shallow)没有相同的效果..手动删除文本元素可能会有用。 - user2864740

-1

使用正则表达式和递归方法,这是一个很容易的任务:

    using System.Xml.Linq;
    public static class Xml
    {
        /// <summary>
        /// Recursive method to shorten all xml end tags starting from a given element, and running through all sub elements 
        /// </summary>
        /// <param name="elem">Starting point element</param>
        public static void ToShortEndTags(this XElement elem)
        {
            if (elem == null) return;

            if (elem.HasElements)
            {
                foreach (var item in elem.Elements()) ToShortEndTags(item);
                return;
            }

            var reduced = Regex.Replace(elem.ToString(), ">[\\s\\n\\r]*</\\w+>", "/>");

            elem.ReplaceWith(XElement.Parse(reduced));
        }
    }

要使用它,输入类似于这样的内容:

    var path = @"C:\SomeFile.xml";
    var xdoc = XDocument.Load(path).Root.ToShortEndTags();

xdoc 现在是从给定路径加载的 XDocument 实例,但其所有符合条件(没有内容)的完整结束标记现在都被缩短了。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接