Я знаю, что затенение членов в реализациях класса может привести к ситуациям, когда «неправильный» член может быть вызван в зависимости от того, как я произвел приведение своих экземпляров, но с интерфейсами я не вижу, что это может быть проблемой, и я нахожу Я часто пишу такие интерфейсы:
public interface INode
{
IEnumerable<INode> Children { get; }
}
public interface INode<N> : INode
where N : INode<N>
{
new IEnumerable<N> Children { get; }
}
public interface IAlpha : INode<IAlpha>
{ }
public interface IBeta : INode<IBeta>
{ }
В моем коде есть места, которые знают только о INode
, поэтому дети также должны быть типа INode
.
В других местах я хочу знать о конкретных типах - в реализации моего примера IAlpha
& IBeta
интерфейсов я хочу, чтобы дочерние элементы были напечатаны так же, как и их родительские.
Итак, я реализую класс NodeBase
следующим образом:
public abstract class NodeBase<N> : INode<N>
where N : INode<N>
{
protected readonly List<N> _children = new List<N>();
public IEnumerable<N> Children
{
get { return _children.AsEnumerable(); }
}
IEnumerable<INode> INode.Children
{
get { return this.Children.Cast<INode>(); }
}
}
В реальной реализации нет теней, только в интерфейсах.
Отдельные экземпляры IAlpha
& IBeta
выглядят так:
public class Alpha : NodeBase<Alpha>, IAlpha
{
IEnumerable<IAlpha> INode<IAlpha>.Children
{
get { return this.Children.Cast<IAlpha>(); }
}
}
public class Beta : NodeBase<Beta>, IBeta
{
IEnumerable<IBeta> INode<IBeta>.Children
{
get { return this.Children.Cast<IBeta>(); }
}
}
Опять же, в реализациях нет теней.
Теперь я могу получить доступ к следующим типам:
var alpha = new Alpha();
var beta = new Beta();
var alphaAsIAlpha = alpha as IAlpha;
var betaAsIBeta = beta as IBeta;
var alphaAsINode = alpha as INode;
var betaAsINode = beta as INode;
var alphaAsINodeAlpha = alpha as INode<Alpha>;
var betaAsINodeBeta = beta as INode<Beta>;
var alphaAsINodeIAlpha = alpha as INode<IAlpha>;
var betaAsINodeIBeta = beta as INode<IBeta>;
var alphaAsNodeBaseAlpha = alpha as NodeBase<Alpha>;
var betaAsNodeBaseBeta = beta as NodeBase<Beta>;
У каждой из этих переменных теперь есть правильная коллекция Children
строго типа.
Итак, мои вопросы просты. Является ли слежка за элементами интерфейса, использующими этот тип шаблона, хорошей, плохой или уродливой? И почему?