Я написал класс, который позволяет производному определять, какие из его свойств могут быть загружены с отложенной загрузкой. Код:
public abstract class SelfHydratingEntity<T> : DynamicObject where T : class {
private readonly Dictionary<string, LoadableBackingField> fields;
public SelfHydratingEntity(T original) {
this.Original = original;
this.fields = this.GetBackingFields().ToDictionary(f => f.Name);
}
public T Original { get; private set; }
protected virtual IEnumerable<LoadableBackingField> GetBackingFields() {
yield break;
}
public override bool TryGetMember(GetMemberBinder binder, out object result) {
LoadableBackingField field;
if (this.fields.TryGetValue(binder.Name, out field)) {
result = field.GetValue();
return true;
} else {
var getter = PropertyAccessor.GetGetter(this.Original.GetType(), binder.Name);
result = getter(this.Original);
return true;
}
}
public override bool TrySetMember(SetMemberBinder binder, object value) {
LoadableBackingField field;
if (this.fields.TryGetValue(binder.Name, out field)) {
field.SetValue(value);
return true;
} else {
var setter = PropertyAccessor.GetSetter(this.Original.GetType(), binder.Name);
setter(this.Original, value);
return true;
}
}
}
И производный класс:
public class SelfHydratingPerson : SelfHydratingEntity<IPerson> {
private readonly IDataRepository dataRepository;
public SelfHydratingDerivate(IDataRepository dataRepository, IPerson person)
: base(person) {
this.dataRepository = dataRepository
}
protected override IEnumerable<LoadableBackingField> GetBackingFields() {
yield return new LoadableBackingField("Address", () => this.dataRepository.Addresses.Get(this.Original.AddressID));
}
}
Это прекрасно работает для получения и настройки значений свойств, но я получаю либо RuntimeBinderException, когда я неявно приводил, либо InvalidCastException с явно приведенным SelfHydratingEntity обратно в T.
Я знаю, что вы можете переопределить метод DynamicObject.TryConvert, но мне интересно, что именно точно поместить в этот метод. Сегодня я много читал о типизации утки и пробовал несколько библиотек, но ни одна из них не подходит для этого конкретного сценария. Все библиотеки, которые я пробовал сегодня, генерируют класс-оболочку, используя Reflection.Emit, который вызывает методы «get_» и «set_» и, естественно, использует отражение, чтобы найти эти методы в обернутом экземпляре. Конечно, в SelfHydratingEntity не определены методы "get_" и "set_".
Итак, мне интересно, возможно ли такое вообще. Есть ли способ привести экземпляр SelfHydratingEntity к T? Я ищу что-то вроде этого:
var original = GetOriginalPerson();
dynamic person = new SelfHydratingPerson(new DataRepository(), original);
string name = person.Name; // Gets property value on original
var address = person.Address; // Gets property value using LoadableBackingField registration
var iPerson = (IPerson)person;
- or -
var iPerson = DuckType.As<IPerson>(person);