Могу ли я сериализовать ExpandoObject в .NET 4? - PullRequest
27 голосов
/ 31 января 2011

Я пытаюсь использовать System.Dynamic.ExpandoObject, чтобы я мог динамически создавать свойства во время выполнения. Позже мне нужно передать экземпляр этого объекта, а используемый механизм требует сериализации.

Конечно, когда я пытаюсь сериализовать свой динамический объект, я получаю исключение:

System.Runtime.Serialization.SerializationException не обработано.

Тип 'System.Dynamic.ExpandoObject' в сборке 'System.Core, версия = 4.0.0.0, Culture = нейтральный, PublicKeyToken = b77a5c561934e089' не помечен как сериализуемый.

Могу ли я сериализовать ExpandoObject? Есть ли другой подход к созданию динамического объекта, который сериализуем? Возможно, с помощью оболочки DynamicObject ?

Я создал очень простой пример Windows Forms для дублирования ошибки:

using System;
using System.Windows.Forms;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;
using System.Dynamic;

namespace DynamicTest
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {            
            dynamic dynamicContext = new ExpandoObject();
            dynamicContext.Greeting = "Hello";

            IFormatter formatter = new BinaryFormatter();
            Stream stream = new FileStream("MyFile.bin", FileMode.Create,
                                           FileAccess.Write, FileShare.None);
            formatter.Serialize(stream, dynamicContext);
            stream.Close();
        }
    }
}

Ответы [ 3 ]

20 голосов
/ 01 февраля 2011

Я не могу сериализовать ExpandoObject, но я могу вручную сериализовать DynamicObject. Поэтому, используя методы DynamicObject TryGetMember / TrySetMember и реализуя ISerializable, я могу решить мою проблему, заключающуюся в том, чтобы действительно сериализовать динамический объект.

В моем простом тестовом приложении я реализовал следующее:

using System;
using System.Windows.Forms;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;
using System.Collections.Generic;
using System.Dynamic;
using System.Security.Permissions;

namespace DynamicTest
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {            
            dynamic dynamicContext = new DynamicContext();
            dynamicContext.Greeting = "Hello";
            this.Text = dynamicContext.Greeting;

            IFormatter formatter = new BinaryFormatter();
            Stream stream = new FileStream("MyFile.bin", FileMode.Create, FileAccess.Write, FileShare.None);
            formatter.Serialize(stream, dynamicContext);
            stream.Close();
        }
    }

    [Serializable]
    public class DynamicContext : DynamicObject, ISerializable
    {
        private Dictionary<string, object> dynamicContext = new Dictionary<string, object>();

        public override bool TryGetMember(GetMemberBinder binder, out object result)
        {
            return (dynamicContext.TryGetValue(binder.Name, out result));
        }

        public override bool TrySetMember(SetMemberBinder binder, object value)
        {
            dynamicContext.Add(binder.Name, value);
            return true;
        }

        [SecurityPermissionAttribute(SecurityAction.Demand, SerializationFormatter = true)]
        public virtual void GetObjectData(SerializationInfo info, StreamingContext context)
        {
            foreach (KeyValuePair<string, object> kvp in dynamicContext)
            {
                info.AddValue(kvp.Key, kvp.Value);
            }
        }

        public DynamicContext()
        {
        }

        protected DynamicContext(SerializationInfo info, StreamingContext context)
        {
            // TODO: validate inputs before deserializing. See http://msdn.microsoft.com/en-us/library/ty01x675(VS.80).aspx
            foreach (SerializationEntry entry in info)
            {
                dynamicContext.Add(entry.Name, entry.Value);
            }
        }

    }
}

и Почему в SerializationInfo нет методов TryGetValue? содержит недостающую часть головоломки, чтобы упростить ее.

9 голосов
/ 31 января 2011

ExpandoObject реализует IDictionary<string, object>, например ::1003 *

class Test
{
    static void Main()
    {
        dynamic e = new ExpandoObject();
        e.Name = "Hello";

        IDictionary<string, object> dict = (IDictionary<string, object>)e;

        foreach (var key in dict.Keys)
        {
            Console.WriteLine(key);
        }

        dict.Add("Test", "Something");

        Console.WriteLine(e.Test);

        Console.ReadKey();
    }
}

Вы можете записать содержимое словаря в файл и затем создать новый ExpandoObject посредством десериализации, вернуть его обратно в словарь и записать свойства обратно в?

8 голосов
/ 27 марта 2011

Возможно, немного поздно, чтобы ответить, но я использую jsonFx для сериализации и десериализации expandoObjects, и это работает очень хорошо:

сериализации:

dim XMLwriter As New JsonFx.Xml.XmlWriter
dim serializedExpando as string =XMLwriter.Write(obj)

десериализация

dim XMLreader As New JsonFx.Xml.XmlReader
Dim obj As ExpandoObject = XMLreader.Read(Str)
...