如果您愿意,您可以在运行时使用 Reflection.Emit 创建一个类。一种可能的方法是让您执行以下操作。
您使用这个类来执行整个过程。使用您希望属性的适当类型关闭类型 T。在您的情况下,您将使用类型字符串。
class ClassFromList<T>
{
public object Instance { get; set; }
public ClassFromList(List<string> list)
{
var type = new DynamicClass().Builder;
list.ForEach(item => new DynamicProperty<T>(type, item));
Instance = Activator.CreateInstance(type.CreateType());
}
}
在上面提到的类中,这个类将为您提供 TypeBuilder 类型的属性,因此您可以在运行时创建一个类。
class DynamicClass
{
public TypeBuilder Builder { get; set; }
public DynamicClass()
{
var name = new AssemblyName("DynamicAssembly");
var assembly = AppDomain.CurrentDomain.DefineDynamicAssembly(name, AssemblyBuilderAccess.Run);
var module = assembly.DefineDynamicModule("DynamicModule");
Builder = module.DefineType("DynamicClass", TypeAttributes.Public);
}
}
另一方面,以下类将创建您希望您的类为您实现的属性。
class DynamicProperty<T>
{
public DynamicProperty(TypeBuilder builder, string name)
{
var field = builder.DefineField("_" + name, typeof(T), FieldAttributes.Private);
var property = builder.DefineProperty(name, PropertyAttributes.None, typeof(T), new Type[0]);
var getter = builder.DefineMethod("get_" + name, MethodAttributes.Public | MethodAttributes.SpecialName, typeof(T), new Type[0]);
var setter = builder.DefineMethod("set_" + name, MethodAttributes.Public | MethodAttributes.SpecialName, null, new Type[] { typeof(T) });
var getGenerator = getter.GetILGenerator();
var setGenerator = setter.GetILGenerator();
getGenerator.Emit(OpCodes.Ldarg_0);
getGenerator.Emit(OpCodes.Ldfld, field);
getGenerator.Emit(OpCodes.Ret);
setGenerator.Emit(OpCodes.Ldarg_0);
setGenerator.Emit(OpCodes.Ldarg_1);
setGenerator.Emit(OpCodes.Stfld, field);
setGenerator.Emit(OpCodes.Ret);
property.SetGetMethod(getter);
property.SetSetMethod(setter);
}
}
现在您只需要测试整个过程。下面的代码将创建 3 个名为“a”、“b”、“c”的字符串类型的属性,分别向它们添加值“value1”、“value2”、“value3”,创建新类的实例,并显示所有属性,它们的类型和值给你。
class Program
{
static void Main()
{
var list = new List<string>() { "a", "b", "c" };
var instance = new ClassFromList<string>(list).Instance;
instance.GetType().GetProperty("a").SetValue(instance, "value1", null);
instance.GetType().GetProperty("b").SetValue(instance, "value2", null);
instance.GetType().GetProperty("c").SetValue(instance, "value3", null);
instance.GetType()
.GetProperties()
.ToList()
.ForEach(x => Console.WriteLine(x.Name + " " + x.PropertyType + " " + x.GetValue(instance, null)));
Console.Read();
}
}
编辑:如果你真的需要你的类中的字段,正如我刚刚注意到的那样,你应该使用 DynamicField 类而不是下面的 DynamicProperty 类。但请注意,对于对象之间的数据传输,属性优先于字段。
class DynamicField<T>
{
public DynamicField(TypeBuilder builder, string name)
{
var field = builder.DefineField(name, typeof(T), FieldAttributes.Public);
}
}