質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.51%
C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

.NET Framework

.NET Framework は、Microsoft Windowsのオペレーティングシステムのために開発されたソフトウェア開発環境/実行環境です。多くのプログラミング言語をサポートしています。

Q&A

1回答

4054閲覧

C#でジェネリックなデリゲートを引数とするメソッドについて その2

退会済みユーザー

退会済みユーザー

総合スコア0

C#

C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

.NET Framework

.NET Framework は、Microsoft Windowsのオペレーティングシステムのために開発されたソフトウェア開発環境/実行環境です。多くのプログラミング言語をサポートしています。

0グッド

1クリップ

投稿2017/08/24 10:04

お世話になります。

こちらで一度質問したのですが、質問内容が私のしたかったことと異なっていたため、改めて質問をさせて頂きます。

うまく言葉にするのが難しいので、コードを載せさせていただきます。

概要としては、

  • Json形式でシリアライズ/デシリアライズクラスと、XML形式でシリアライズ/デシリアライズクラスを用意します。
  • それらはIConvertWrapperというインターフェイスをimplementし、ConvertFactoryからEnumを引数としてシリアライズ/デシリアライズメソッドを呼び出せるようにします。
  • クライアント(ここではTestメソッド)では、Enumとクラス型を指定して、シリアライズ/デシリアライズできるようにします。

ここで、質問の本質はシリアライズ/デシリアライズではなく、
ジェネリック型を返すデリゲートを引数としたメソッドをどのようにコーディングしたらよいのか?です。
(ただシリアライズ/デシリアライズするだけならこのような遠回りのことはしないのですが、今回は例えばコールバックでシリアライズ/デシリアライズを行う必要があり、直接具象クラスにアクセスできないという制約があります)

↓のサンプルでは、デシリアライズした際にobject型となるため、デシリアライズがうまくいきません。

どなたかご教授お願い致します。

C#

1using System; 2using System.Collections.Generic; 3using System.Text; 4 5namespace Convert 6{ 7 public delegate string SerializeFunction<T>(T obj) where T : class; 8 public delegate T DeserializeFunction<T>(string message) where T : class; 9 10 public enum MessageFormat 11 { 12 Json = 1, 13 Xml, 14 } 15 16 interface IConvertWrapper 17 { 18 string Serialize<T>(T obj) where T : class; 19 T Deserialize<T>(string message) where T : class; 20 } 21} 22

C#

1using System; 2using System.Collections.Generic; 3using System.Text; 4using Newtonsoft.Json; 5using Newtonsoft.Json.Serialization; 6 7namespace Convert 8{ 9 // クラス内のロジックは質問の本質ではありません 10 class JsonConvertWrapper : IConvertWrapper 11 { 12 public string Serialize<T>(T obj) where T : class 13 { 14 var setting = new JsonSerializerSettings { MissingMemberHandling = MissingMemberHandling.Ignore, NullValueHandling = NullValueHandling.Ignore, Formatting = Formatting.Indented, ContractResolver = new CamelCasePropertyNamesContractResolver() }; 15 return JsonConvert.SerializeObject(obj, setting); 16 } 17 18 public T Deserialize<T>(string message) where T : class 19 { 20 return JsonConvert.DeserializeObject<T>(message); 21 } 22 } 23} 24

C#

1using System; 2using System.Collections.Generic; 3using System.IO; 4using System.Text; 5using System.Xml; 6using System.Xml.Serialization; 7 8namespace Convert 9{ 10 // クラス内のロジックは質問の本質ではありません 11 class XmlConvertWrapper : IConvertWrapper 12 { 13 public string Serialize<T>(T obj) where T : class 14 { 15 var sb = new StringBuilder(); 16 17 using (var writer = XmlWriter.Create(sb)) 18 { 19 var serializer = new XmlSerializer(obj.GetType()); 20 serializer.Serialize(writer, obj); 21 } 22 23 return sb.ToString(); 24 } 25 26 public T Deserialize<T>(string message) where T : class 27 { 28 using(var reader = XmlReader.Create(new StringReader(message))) 29 { 30 var serializer = new XmlSerializer(typeof(T)); 31 T obj = serializer.Deserialize(reader) as T; 32 return obj; 33 } 34 35 } 36 } 37} 38

C#

1using System; 2using System.Collections.Generic; 3using System.Text; 4 5namespace Convert 6{ 7 class ConvertFactory 8 { 9 static Dictionary<MessageFormat, IConvertWrapper> converters = new Dictionary<MessageFormat, IConvertWrapper>(); 10 11 static ConvertFactory() 12 { 13 converters.Add(MessageFormat.Json, new JsonConvertWrapper()); 14 converters.Add(MessageFormat.Xml, new XmlConvertWrapper()); 15 } 16 17 // ↓ここは自分でもよくわかりません。動的型付けになっている? 18 static public SerializeFunction<object> GetSerializeFunction(MessageFormat messageFormat) 19 { 20 return converters[messageFormat].Serialize<object>; 21 } 22 23 // ↓ここは自分でもよくわかりません。動的型付けになっている? 24 static public DeserializeFunction<object> GetDeserializeFunction(MessageFormat messageFormat) 25 { 26 return converters[messageFormat].Deserialize<object>; 27 } 28 } 29} 30

C#

1class Foo() 2{ 3 public int X {get; set;} 4 public string Y {get; set;} 5} 6 7void Test() 8{ 9 foreach (MessageFormat format in Enum.GetValues(typeof(MessageFormat))) 10 { 11 var foo1 = new Foo{X=100, Y="test"}; 12 var txt = ConvertFactory.GetSerializeFunction(format)(foo1); 13 14 var foo2 = ConvertFactory.GetDeserializeFunction(format)(txt); 15 // ↑ここで foo2をでfoo1の情報でシリアライズしたいが、このままではfoo2はobject型のままである 16 } 17}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ConvertFactoryクラスをジェネリッククラスにするのではダメなのでしょうか?

C#

1public class ConvertFactory<T> where T : class 2{ 3 static Dictionary<MessageFormat, IConvertWrapper> converters = new Dictionary<MessageFormat, IConvertWrapper>(); 4 5 static ConvertFactory() 6 { 7 converters.Add(MessageFormat.Json, new JsonConvertWrapper()); 8 converters.Add(MessageFormat.Xml, new XmlConvertWrapper()); 9 } 10 11 // ↓ここは自分でもよくわかりません。動的型付けになっている? 12 static public SerializeFunction<T> GetSerializeFunction(MessageFormat messageFormat) 13 { 14 return converters[messageFormat].Serialize<T>; 15 } 16 17 // ↓ここは自分でもよくわかりません。動的型付けになっている? 18 static public DeserializeFunction<T> GetDeserializeFunction(MessageFormat messageFormat) 19 { 20 return converters[messageFormat].Deserialize<T>; 21 } 22}

C#

1var foo1 = new Foo { X = 100, Y = "test" }; 2var txt = Convert.ConvertFactory<Foo>.GetSerializeFunction(format)(foo1); 3 4var foo2 = Convert.ConvertFactory<Foo>.GetDeserializeFunction(format)(txt);

このままだとXmlConvertWrapperのDeserializeでコケますが。。

見当違いだったらゴメンなさい。

投稿2017/08/24 11:12

編集2017/08/24 11:13
ebiryo

総合スコア797

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.51%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問