Как добавить новый элемент в сериализованные данные XML без десериализации старых данных? - PullRequest
3 голосов
/ 20 февраля 2012

Обычно я использую следующий код для сериализации объекта в файл XML.Каждый день у меня есть около 100-1000 новых элементов, которые будут добавляться в этот список в разные периоды времени.

var xmlSerializer = new XmlSerializer(typeof(List<TestModel>));
xmlSerializer.Serialize(stream, list);

Как добавить новый элемент в сериализованные данные XML без десериализации старых данных?

Спасибо

Ответы [ 3 ]

5 голосов
/ 20 февраля 2012

Вы можете сериализовать objecto в память и добавить в существующий файл. Также обратите внимание на статью MS Эффективные методы изменения больших файлов XML , в которой показаны две технологии, применимые в вашей ситуации.

3 голосов
/ 20 февраля 2012

Нет способа сделать то, что вы хотите, используя только XmlSerializer, который я могу придумать, но с небольшой дополнительной работой это возможно.

Простой подход к этому -сериализовать список для первого элемента (ов) дня - как делает ваш существующий код.Когда поступают новые данные, теперь вы можете открыть сохраненный xml с помощью XmlDocument и добавить сериализацию по одному элементу за раз.

Следует отметить, что если результирующий xml очень большой,XmlDocument может стать очень большим (и может быть медленным или даже вызывать исключения OutOfMemoryException, как отмечает Павел Киментс в комментарии), в этом случае вы можете захотеть изучить XmlReader и XmlWriter для последовательного добавления xml.Однако общий подход остался бы прежним (open-> serialize вашего нового элемента-> добавляют сгенерированный xml-> resave)

[ EDIT - измененный пример кода для отображения цепочки XmlReader/ XmlWriter, а не XmlDocument подход]

Что-то вроде этого:

public static void AppendToXml(
    Stream xmlSource,             // your existing xml - could be from a file, etc
    Stream updatedXmlDestination, // your target xml, could be a different file
    string rootElementName,       // the root element name of your list, e.g. TestModels
    TestModel itemToAppend)       // the item to append
{
    var writerSettings = new XmlWriterSettings {Indent = true, IndentChars = " " };
    using (var reader = XmlReader.Create(xmlSource))
    using (var writer = XmlWriter.Create(updatedXmlDestination, writerSettings))
    {
        while (reader.Read())
        {
            switch (reader.NodeType)
            {
                case XmlNodeType.XmlDeclaration:
                    break;
                case XmlNodeType.Element:
                    writer.WriteStartElement(reader.Prefix, reader.LocalName, reader.NamespaceURI);
                    if (reader.HasAttributes)
                    {
                        while (reader.MoveToNextAttribute())
                        {
                            writer.WriteAttributeString(reader.Prefix, reader.LocalName, reader.NamespaceURI, reader.Value);
                        }
                    }
                    if (reader.IsEmptyElement) 
                        writer.WriteEndElement();
                    break;
                case XmlNodeType.EndElement:
                    if (reader.Name == rootElementName)
                    {
                        var serializer = new XmlSerializer(typeof(TestModel));
                        var ns = new XmlSerializerNamespaces();
                        ns.Add("", "");
                        serializer.Serialize(writer, itemToAppend, ns);
                    }
                    writer.WriteEndElement();
                    break;
                case XmlNodeType.Text:
                    writer.WriteRaw(SecurityElement.Escape(reader.Value));
                    break;
                case XmlNodeType.CDATA:
                    writer.WriteCData(reader.Value);
                    break;
            }
        }
    }
}

Примечание: вы можете добавить поддержку других типов узлов(здесь для краткости опущено), такие как пробелы, комментарии, инструкции обработки и т. д. Все они следуют тому же шаблону, что и CDATA выше: введите регистр, вызовите соответствующий метод записи.

При таком обновленном подходе -у вас никогда не бывает больше, чем небольшое количество памяти в любой момент времени.

0 голосов
/ 20 февраля 2012

Я не думаю, что это возможно. Вы хотите выполнить произвольный доступ к блоку данных, который сериализован и десериализован , поэтому к нему необходимо обращаться последовательно. Может быть, вы можете изменить непосредственно XML-документ, что будет быстрее, но вы потеряете возможности использования сериализованного / десериализованного дерева объектов, которым гораздо проще управлять (добавлять / удалять объекты, ...)

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...