Идентификация производных типов из списка объектов базового класса - PullRequest
5 голосов
/ 16 января 2010

Это может показаться "домашним заданием" и / или тривиальным, но это для реальных деловых целей; это просто самый простой способ объяснить, что я концептуально пытаюсь сделать.

Предположим, у меня есть класс Animal и некоторые другие классы (Bird, Cat, Kangaroo). Каждый из них наследуется от Animal.

Животное может выглядеть так:

public class Animal
{
    public Animal()
    {

    }

    public string Name { get; set; }
    public string AnimalType { get; set; }
    public List<Animal> Friends { get; set; }
}

Кенгуру может выглядеть так:

public class Kangaroo : Animal
{
    public Kangaroo()
    {

    }

    public int TailLengthInches { get; set; }
}

Предположим, у кенгуру есть два друга, птица и кошка. Как я могу добавить их в список друзей «Животных» (типа «Животное»), но сохранить возможность доступа к свойствам производных классов? (например, мне все еще нужно уметь работать со свойствами, специфичными для данного типа животных, например, для FeatherColor для птицы, даже если они будут считаться просто «животными».

Причина, по которой я пытаюсь это сделать, заключается в том, что, когда я позже получаю список «друзей» животного, очень важно знать, какой тип животного есть у друга. В зависимости от типа животного (будь то птица, кошка и т. Д.), Я хочу сделать что-то другое (на самом деле, показать другой шаблон на странице ASP.NET, но мне конкретно это не нужно).

Полагаю, это сводится к следующему ... если у меня есть список Animal, откуда я знаю, каким был производный тип для каждого из объектов в списке, чтобы я мог привести их обратно к их производному типу или сделать что-то еще, что позволяет мне получить к определенным, различным свойствам каждого производного типа?

Спасибо!

Ответы [ 5 ]

8 голосов
/ 16 января 2010

Вы можете использовать LINQ для фильтрации по типу, который вы ищете

animals.Friends = new List<Animal> { new Kangaroo(), new Bird() };
foreach ( var kangaroo in animals.Friends.OfType<Kangaroo>()) {
  kangaroo.Hop();
}
2 голосов
/ 16 января 2010

Попробуйте разыграть это:

var friendBird = friend as Bird;
if (friendBird != null)
{
    // It's a bird, so do something with it
)
1 голос
/ 16 января 2010

Вы можете использовать typeof, чтобы получить фактический тип, или использовать операторы is или as, чтобы проверить и при желании привести:

foreach (Animal a in Friends)
{
  Kangaroo k = a as Kangaroo;
  if (a != null)
  {
    // it's a kangaroo
    k.JumpAround();
  }
  Ostrich o = a as Ostrich;
  if (o != null)
  {
    o.StickHeadInSand();
  }
}

(я не использовал здесь else предложения, которые могут вызвать ошибки, если вы тестируете типы, которые могут быть в иерархии, например, если у вас есть предложения Kangaroo и Marsupial, вы, вероятно, НЕ хотите, чтобы оба выполнено!)

Вы также можете сделать это с оператором is, который выглядит лучше, но заставит FxCop плакать:

foreach (Animal a in Friends)
{
  if (a is Kangaroo)
  {
    ((Kangaroo)a).JumpAround();
  }
}

В общем, между прочим, вы, вероятно, хотите найти возможность поместить виртуальный метод в Animal и реализовать этот метод полиморфно, вместо того, чтобы иметь операторы-монстры, которые приводят к хрупкой связи с определенным набором производные классы.

0 голосов
/ 16 января 2010
animals.Friends = new List<Animal> { new Kangaroo(), new Bird() };
foreach (Animal animal in animals.Friends)
{
if (animal is Kangaroo)
Console.WriteLine("Kangaroo");
else if (animal is Bird)
Console.WriteLine("Bird");
}
0 голосов
/ 16 января 2010

Используйте ключевое слово "is".

if (животное - птица) ...

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...