まず第一に、私の問題を喜んで見てくれる人に感謝します。私は Nancyfx を初めて使用し、JSON ペイロードを動的に作成されたクラスにバインドしようとすると問題が発生します。この投稿のコードに従って動的にクラスを作成しました - C# でクラスを動的に作成します
これは、動的クラスを作成するコードのブロックです。これは、本質的に、ダニエルズによって提供されたコードのカットアンドペーストであることを認めます
public static Type CompileResultType(List<Metadata> metadata)
{
TypeBuilder tb = GetTypeBuilder();
ConstructorBuilder constructor = tb.DefineDefaultConstructor(MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.RTSpecialName);
foreach (var field in metadata)
{
if(field.datatype == "String")
{
Type datatype = Type.GetType("System.String");
CreateProperty(tb, field.columnname, datatype);
}
if (field.datatype == "int")
{
Type datatype = Type.GetType("System.Int32");
CreateProperty(tb, field.columnname, datatype);
}
if(field.datatype == "datetime")
{
Type datatype = Type.GetType("System.DateTime");
CreateProperty(tb, field.columnname, datatype);
}
}
Type objectType = tb.CreateType();
return objectType;
}
private static TypeBuilder GetTypeBuilder()
{
var typeSignature = "MyDynamicType";
var an = new AssemblyName(typeSignature);
AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(an, AssemblyBuilderAccess.Run);
ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule("MainModule");
TypeBuilder tb;
tb = moduleBuilder.DefineType(typeSignature, TypeAttributes.Public | TypeAttributes.Class | TypeAttributes.AutoClass | TypeAttributes.AnsiClass | TypeAttributes.BeforeFieldInit | TypeAttributes.AutoLayout, null);
return tb;
}
private static void CreateProperty(TypeBuilder tb, string propertyName, Type propertyType)
{
FieldBuilder fieldBuilder = tb.DefineField(propertyName, propertyType, FieldAttributes.Public); //changed field attributes from private to public
PropertyBuilder propertyBuilder = tb.DefineProperty(propertyName, System.Reflection.PropertyAttributes.None, propertyType, null);
MethodBuilder getPropMethdBldr = tb.DefineMethod("get_" + propertyName, MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig, propertyType, Type.EmptyTypes); //changed get_ to get
ILGenerator getIl = getPropMethdBldr.GetILGenerator();
getIl.Emit(OpCodes.Ldarg_0);
getIl.Emit(OpCodes.Ldfld, fieldBuilder);
getIl.Emit(OpCodes.Ret);
MethodBuilder setPropMthdBldr = tb.DefineMethod("set_" + propertyName, MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig, null, new[] { propertyType }); //changed set_to set
ILGenerator setIl = setPropMthdBldr.GetILGenerator();
Label modifyProperty = setIl.DefineLabel();
Label exitSet = setIl.DefineLabel();
setIl.MarkLabel(modifyProperty);
setIl.Emit(OpCodes.Ldarg_0);
setIl.Emit(OpCodes.Ldarg_1);
setIl.Emit(OpCodes.Stfld, fieldBuilder);
setIl.Emit(OpCodes.Nop);
setIl.MarkLabel(exitSet);
setIl.Emit(OpCodes.Ret);
propertyBuilder.SetGetMethod(getPropMethdBldr);
propertyBuilder.SetGetMethod(setPropMthdBldr);
}
//List<Metadata> metadata is a Dictionary
次に、これを実行してバインドします
var myType = CompileResultType(metadata);
var myObject = Activator.CreateInstance(myType);
var b = this.BindTo(myObject);
データを収集するためだけにモデルが多すぎるため、Nancy の通常のモデル クラスを使用することはできません。
//usual model class
public Class PayloadModel
{
public string firstName {get; set;}
public string lastName {get; set;}
}
クラスを動的に作成する必要がある理由は、受け取る JSON ペイロードが 1 フィールドから 30 フィールドまでさまざまであるためです。もう 1 つの理由は、できるだけ柔軟に保ちたいからです。そして、上記のクラスを持つということは、そのクラスにあるものに縛られていることを意味します。どんなアドバイスや指示も大いに役立ちます!