删除结尾元素的空间?

时间:2011-01-05 03:25:56

标签: c# xml

目前我使用XML来存储许多数据,在创建这些XML文件时,我希望将其大小减少到最小值。

我怎样才能覆盖XmlWriter函数(WriteEndElement),而不是像以下那样保存它:

<thisElement someAttribute="blabla" />

它会像以下一样保存:

<thisElement someAttribute="blabla"/>   

更新

我试图找到一种方法来使用:

来实现这一点
public override void WriteEndElement()

但是我不能用当前的WriteEndElement函数知道我必须要改变什么以及它是否可能。

1 个答案:

答案 0 :(得分:1)

如果没有重写代码的重要部分,我担心这是不可能的。该空间在内部类中是硬编码的,不可配置。

,例如,内部XmlEncodedRawTextWriter.WriteEndElement()方法的代码段。

internal override void WriteEndElement(string prefix, string localName, string ns)
{
    // snip...
    else
    {
        this.bufPos--;
        this.bufChars[this.bufPos++] = ' '; // the space is hard coded
        this.bufChars[this.bufPos++] = '/';
        this.bufChars[this.bufPos++] = '>';
    }
}

我能想到的一些选项是解析输出的XML以搜索结束标记以手动删除空格,实现自己的XML编写器以使其不包含此空间,或者编写包装类在写入结束元素时使用反射来修改内部缓冲区。


这是一种可以做到这一点的扩展方法。请注意,这不是便携式的。它也不能保证适用于所有情况,尽管它似乎适用于简单的情况。我不认为这里所做的会破坏作家的状态,但使用风险自负

public static class XmlWriterExtensions
{
    private static readonly Func<XmlWriter, object> get_writer;
    private static readonly Func<object, char[]> get_bufChars;
    private static readonly Func<object, int> get_bufPos;
    private static readonly Action<object, int> set_bufPos;

    static XmlWriterExtensions()
    {
        var asm = Assembly.GetAssembly(typeof(XmlWriter));
        var xmlWellFormedWriterType = asm.GetType("System.Xml.XmlWellFormedWriter");
        var flags = BindingFlags.NonPublic | BindingFlags.Instance;
        var writerField = xmlWellFormedWriterType.GetField("writer", flags);
        get_writer = w => writerField.GetValue(w);
        var xmlEncodedRawTextWriterType = asm.GetType("System.Xml.XmlEncodedRawTextWriter");
        var bufCharsField = xmlEncodedRawTextWriterType.GetField("bufChars", flags);
        var bufPosField = xmlEncodedRawTextWriterType.GetField("bufPos", flags);
        get_bufChars = w => (char[])bufCharsField.GetValue(w);
        get_bufPos = w => (int)bufPosField.GetValue(w);
        set_bufPos = (w, i) => bufPosField.SetValue(w, i);

    }

    public static void TrimElementEnd(this XmlWriter writer)
    {
        var internalWriter = get_writer(writer);
        char[] bufChars = get_bufChars(internalWriter);
        int bufPos = get_bufPos(internalWriter);
        if (bufPos > 3 && bufChars[bufPos - 3] == ' ' && bufChars[bufPos - 2] == '/' && bufChars[bufPos - 1] == '>')
        {
            bufChars[bufPos - 3] = '/';
            bufChars[bufPos - 2] = '>';
            bufPos--;
            set_bufPos(internalWriter, bufPos);
        }
    }
}

// usage:
Console.OutputEncoding = Encoding.UTF8;
using (var writer = XmlWriter.Create(Console.Out))
{
    writer.WriteStartElement("Foo");
    writer.WriteElementString("Bar", null);
    writer.TrimElementEnd();
    writer.WriteElementString("Baz", null);
    writer.WriteEndElement();
}

<?xml version="1.0" encoding="utf-8"?><Foo><Bar/><Baz /></Foo>