我有两个这样的类:
public abstract class MyBase
{
protected MyBase(){
Initialize();
}
protected IDictionary<string,string> _data;
private void Initialize() {
// Use Reflection to get all properties
// of the derived class (e.g., call new MyDerived() then
// I want to know the names "Hello" and "ID" here
var data = GetDataFromBackend(propertyNamesFromDerived);
_data = data;
}
}
public class MyConcrete : MyBase
{
public MyConcrete(){
// Possibly use Reflection here
Hello = _data["Hello"];
ID = new Guid(data["ID"]);
}
public string Hello {get;set;}
public Guid ID {get; set;}
}
如您所见,我希望基类的构造函数了解我正在实例化的派生类的属性。
现在,这似乎是一个巨大的代码味道,所以让我给出更多关于我的意图的背景,也许有更好的方法。
我有一个存储键/值对的后端系统,本质上是一个 Dictionary<string,string>
.我想以一种人们可以创建其属性为后端系统的键的类的方式抽象出使用这个后端系统。当他们构建这个对象时,它会自动从该系统加载数据并将所有变量初始化给它。
换句话说,我刚刚重新发明了序列化,只是我不控制后端系统,只是让它变得非常轻松。我不希望调用者必须调用 Initialize()
在构建对象之后,因为在 100% 的情况下你必须在构建之后初始化它。
我不想将初始化代码移动到派生类中,除了字符串到业务对象的转换。
我必须使用工厂吗?或者在基类构造函数中查看派生类的属性 names 是否安全? (不关心它们的值和它们没有被初始化,只需要名字)。
或者是否有更好的方法在字符串字典和具体业务对象之间提供外观?
编辑:这是 .net 3.5,所以没有 System.Dynamic 会使这变得微不足道:(
编辑 2:在查看答案并进一步思考之后,我想我的问题现在真的可以归结为:正在从一个基本构造函数中调用 GetType().GetProperties()为了获取属性的名称,以及它们是否装饰有某个属性安全?
最佳答案
等等,让我们在这里停一下,然后正确地执行此操作。 这不应该是 MyBase
的责任。
所以你写了一个类来为你管理从后端获取东西,然后你在那个类上写了一个类似于
的方法T Get<T>() where T : new()
并且您让 Get
负责从后端读取字典并使用反射来填充 T
的实例。因此,你说
var concrete = foo.Get<MyConcrete>();
这并不难,而且是正确的方法。
顺便说一下,Get
的代码看起来像
T t = new T();
var properties = typeof(T).GetProperties();
foreach(var property in properties) {
property.SetValue(t, dictionary[property.Name], null);
}
return t;
其中 dictionary
是您加载的键/值对。事实证明,有更多优化方法可以做到这一点,但除非它是一个瓶颈,否则我不会担心它。
关于c# - 使用反射从基类构造函数在派生类上设置值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6781478/