我有5个具有相同属性但名称空间不同的类。
例如:
MyNameSpace1.Class1 - FirstName, LastName, Age
MyNameSpace2.Class2 - FirstName, LastName, Age
以此类推
。
所有类都具有相同的属性-例如,名字,姓氏,年龄等。
private void AssignData(int order, string firstName, string lastName)
{
if(order==1)
{
var result=new MyNameSpace1.Class1();
result.FirstName= firstName;
result.LastName=lastName;
}
if(order==2)
{
var result=new MyNameSpace2.Class2();
result.FirstName= firstName;
result.LastName=lastName;
}
}
上面,我只给出了两个属性,在我的实际示例中,有多个属性被分配。
我试图简化为
private void AssignData(int order, string firstName, string lastName)
{
if(order==1)
{
var result= GetInstance<MyNameSpace1.Class1>();
result.FirstName=firstName;
result.LastName=lastName;
}
if(order==2)
{
var result= GetInstance<MyNameSpace2.Class2>();
result.FirstName=firstName; //repeated code , don’t want to use dynamic,
//as I will not know the compile time issues.
result.LastName=lastName;
}
}
似乎,我再次重复了相同的代码,是否有一个选项,我只需要直接分配一次firstName,lastName并创建该类的运行时实例?
private T GetInstance<T>()
{
return Activator.CreateInstance<T>();
}
我想要实现的是
private void AssignData(int order, string firstName, string lastName)
{
var result; //this will not compile, object might, prefer not to use dynamic
if(order==1)
{
result= GetInstance<MyNameSpace1.Class1>();
}
if(order==2)
{
result= GetInstance<MyNameSpace2.Class2>();
}
result.FirstName=firstName;//just do assignments 1 time
result.LastName=lastName;
}
答案 0 :(得分:0)
第一个也是最简单的选择就是使用dynamic
类型:
class MyClass
{
public int MyProperty { get; set; }
}
private dynamic GetInstance<T>()
{
return Activator.CreateInstance<T>();
}
var instance = GetInstance<MyClass>();
instance.MyProperty = 5;
Console.WriteLine(instance.MyProperty);
所以我不建议这样做,因为它被认为不是很出色。
第二个选项将使用这样的反射:
var instance = GetInstance<MyClass>();
typeof(MyClass).GetProperty("MyProperty")
.GetSetMethod()
.Invoke(instance, new object[] { 5}); // sets MyProperty to 5
您可以“缓存” GetSetMethod
的结果并重复使用,因为反射也很慢。
最后一个选择是反射+表达式树(您可以在其中生成“整个”方法来设置所有变量):
var type = typeof(MyClass);
var method = type.GetProperty(nameof(MyClass.MyProperty)).GetSetMethod();
var cls = Expression.Parameter(typeof(MyClass));
var val = Expression.Parameter(typeof(int));
var call = Expression.Call(cls, method, val);
// this action should be cached
var act = Expression.Lambda<Action<MyClass, int>>(call, cls, val).Compile();
var instance = GetInstance<MyClass>();
act(instance, 5); // sets MyProperty to 5
再次不要忘记为类型缓存Expression.Lambda.Compile
的结果。