web-dev-qa-db-ja.com

.NET 4でExpandoObjectをシリアル化できますか?

実行時にプロパティを動的に作成できるように、System.Dynamic.ExpandoObjectを使用しようとしています。後で、このオブジェクトのインスタンスを渡す必要があり、使用されるメカニズムにはシリアル化が必要です。

もちろん、動的オブジェクトをシリアル化しようとすると、例外が発生します。

System.Runtime.Serialization.SerializationExceptionが処理されませんでした。

タイプ 'System.Dynamic.ExpandoObject' in Assembly'System.Core、Version = 4.0.0.0、Culture = neutral、PublicKeyToken = b77a5c561934e089 'はシリアル化可能としてマークされていません。

ExpandoObjectをシリアル化できますか?シリアル化可能な動的オブジェクトを作成する別のアプローチはありますか?おそらく DynamicObject ラッパーを使用していますか?

エラーを再現するために、非常に単純なWindowsフォームの例を作成しました。

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();
        }
    }
}
29
Michael Levy

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メソッドがないのはなぜですか? 単純にするためにパズルのピースが欠けていました。

20
Michael Levy

ExpandoObjectIDictionary<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を作成し、それをディクショナリにキャストして戻し、プロパティを書き戻すことができますか?

9
Matthew Abbott

答えるのが少し遅いかもしれませんが、私は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)
8
alainb