クラス実装でメンバーをシャドウイングすると、インスタンスのキャスト方法によっては「間違った」メンバーが呼び出される可能性があることを知っていますが、インターフェイスでは、これが問題になることはなく、インターフェイスを作成していることに気付きます。このように頻繁に:
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
コレクションが含まれるようになりました。
だから、私の質問は簡単です。この種のパターンを使用したインターフェイスメンバーのシャドウイングは、良いですか、悪いですか、それとも醜いですか?なぜ?