インターフェイスから派生クラスメンバーにアクセスする方法は?
-
21-09-2019 - |
質問
3つのクラスがあります。インターフェイスiProductを実装するスタンプ、レター、区画もあり、独自の機能もあります。
public interface IProduct
{
string Name { get; }
int Quantity { get; set; }
float Amount { get; }
}
public class Stamp : IProduct
{
public string Name { get { return "Stamp"; } }
public int Quantity { get; set; }
public float Amount { get; set; }
public float UnitPrice { get; set; }
}
public class Letter : IProduct
{
public string Name { get { return "Letter"; } }
public int Quantity { get; set; }
public float Amount { get; set; }
public float Weight { get; set; }
public string Destination { get; set; }
}
public class Parcel : IProduct
{
public string Name { get { return "Parcel"; } }
public int Quantity { get; set; }
public float Amount { get; set; }
public float Weight { get; set; }
public string Destination { get; set; }
public int Size { get; set; }
}
public static class ShoppingCart
{
private static List<IProduct> products = new List<IProduct>();
public static List<IProduct> Items { get { return products; } }
}
派生クラスの追加メンバーにからアクセスできないのはなぜ List<IProduct>
?
ShoppingCart.Items.Add(new Stamp { Quantity = 5, UnitPrice = 10, Amount = 50 });
ShoppingCart.Items.Add(new Letter { Destination = "US", Quantity = 1, Weight = 3.5f });
ShoppingCart.Items.Add(new Parcel { Destination = "UK", Quantity = 3, Weight = 4.2f, Size = 5 });
foreach (IProduct product in ShoppingCart.Items)
{
Console.WriteLine("Name: {0}, Quantity: {1}, Amount: {2}", product.Name, product.Quantity, product.Amount);
}
ジェネリックを使用することを考えましたが、その場合、特定の種類の製品ごとに個別のコードを作成する必要があります。
public static class ShoppingCart<T> where T : IProduct
{
private static List<T> items = new List<T>();
public static List<T> Items { get { return items; } }
}
ShoppingCart<Stamp>.Items.Add(new Stamp { Quantity = 5, Amount = 10, UnitPrice = 50 });
ShoppingCart<Letter>.Items.Add(new Letter { Destination = "US", Quantity = 1, Weight = 3.5f });
foreach (Stamp s in ShoppingCart<Stamp>.Items)
{
Console.WriteLine("Name: {0}, Quantity: {1}, Amount: {2}", s.Name, s.Quantity, s.Amount);
}
foreach (Letter l in ShoppingCart<Letter>.Items)
{
Console.WriteLine("Name: {0}, Destination: {1}, Weight: {2}", l.Name, l.Destination, l.Weight);
}
この種の問題のためのデザインパターンはありません。工場パターン?
解決
これは、ショッピングカートに各アイテムをforeachループのiproductとしてキャストしているためです。あなたがする必要があるのは、次のようなものです。
foreach(IProduct product in ShoppingCart.Items)
{
if (product is Stamp)
{
var stamp = product as Stamp;
Console.WriteLine("Name: {0}, Quantity: {1}, Amount: {2}, UnitPrice: {3}", stamp.Name, stamp.Quantity, stamp.Amount, stamp.UnitPrice);
}
else if (product is Letter)
{
var letter = product as Letter;
Console.WriteLine("Name: {0}, Quantity: {1}, Amount: {2}, Weight: {3}, Destination: {4}", letter.Name, letter.Quantity, letter.Amount, letter.Weight, letter.Destination);
}
else if (product is Parcel)
{
var parcel = product as Parcel;
Console.WriteLine("Name: {0}, Quantity: {1}, Amount: {2}, Weight: {3}, Destination: {4}, Size: {5}", parcel.Name, parcel.Quantity, parcel.Amount, parcel.Weight, parcel.Destination, parcel.Size);
}
}
また、不要なプロパティを繰り返しています 名前、数量、および量. 。製品から各クラスを導き出す必要があります。
public class Stamp: Product, IProduct
{
public double UnitPrice { get; set; }
}
public class TransitProduct: Product, IProduct
{
public double Weight { get; set; }
public string Destination { get; set; }
}
public class Letter: TransitProduct, IProduct
{
}
public class Parcel: TransitProduct, IProduct
{
public double Size { get; set; }
}
他のヒント
露出しているだけなので、インターフェイスを実装するクラスの追加メンバーにアクセスすることはできません IProduct
アイテムのリスト。ショッピングカート内の各アイテムの特定のリストタイプをショッピングカートクラスに追加すると、iProductインターフェイスのみを使用する必要があるものに対して、カート内のすべての製品のシーケンスを公開できます。
public class ShoppingCart
{
public IList<Stamp> Stamps { get; }
public IList<Letter> Letters { get; }
public IList<Parcel> Parcels { get; }
public IEnumerable<IProduct> Products
{
get
{
return this.Stamps.Cast<IProduct>()
.Concat(this.Letters.Cast<IProduct>())
.Concat(this.Parcels.Cast<IProduct>());
}
}
}
派生クラスの追加メンバーにからアクセスできないのはなぜ
List<IProduct>
?
それは、 IProduct
インターフェイスは知りません UnitPrice
, Destination
派生クラスのプロパティなど。
インテリジェンスを追加して計算しようとしていますか Amount
派生したクラスオブジェクトの各スタンプ、文字、小包に?
それから、私はあなたが少し再設計して、 デコレーターのデザインパターン.
DerivedClass::Amount()
{
Base::Amount() +
//Amount logic based on derived class
}
派生クラスから追加のメンバーにアクセスできない理由は、リストのインターフェイスを使用しているためです。したがって、そのインターフェイスのプロパティのみにアクセスできるようになります。
あなたを助けるかもしれないパターンは、ダブルディスパッチパターンです。
以下の例:
public interface IHandler
{
void Handle(Stamp stamp);
void Handle(Letter letter);
...
}
public class Handler : IHandler
{
public void Handle(Stamp stamp)
{
// do some specific thing here...
}
public void Handle(Letter letter)
{
// do some specific thing here...
}
...
}
public interface IProduct
{
string Name { get; }
int Quantity { get; set; }
float Amount { get; }
void Handle(IHandler handler);
}
public class Stamp : IProduct
{
public string Name { get { return "Stamp"; } }
public int Quantity { get; set; }
public float Amount { get; set; }
public float UnitPrice { get; set; }
public void Handle(IHandler handler)
{
handler.Handle(this);
}
}
これで、ハンドラーで特定の機能をプログラムできるようになりました - 数量 *単価や宛先ルックアップテーブルなど、何らかの合計価格を計算したいと思います...