Можно ли сериализовать ExpandoObject in.NET 4?

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

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

.Во время выполнения.Сериализация.SerializationException был необработан.

Тип '.Активный.Система ExpandoObject "в сборке".Ядро, Версия=4.0.0.0, культура=нейтральная, 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 ответов


Я не могу сериализовать ExpandoObject, но я могу вручную сериализовать DynamicObject. Поэтому, используя методы Trygetmember / TrySetMember DynamicObject и реализуя 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? был недостающий кусок головоломки, чтобы сохранить его простым.


ExpandoObject осуществляет IDictionary<string, object>, например:

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 через десериализацию, вернуть его в словарь и записать свойства обратно?


возможно, немного поздно отвечать, но я использую 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)