Модераторы: Partizan, gambit
  

Поиск:

Ответ в темуСоздание новой темы Создание опроса
> как указать что член класса должен, реализовывать интерфейс 
:(
    Опции темы
Kefir
Дата 28.6.2007, 12:34 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


«Hakuna Matata»
***


Профиль
Группа: Комодератор
Сообщений: 1878
Регистрация: 25.1.2003
Где: Tampere, Suomi

Репутация: 2
Всего: 87



САБЖ.

если развёртнуто, то допустим имеем класс
Код

class Foo {
public List<object> Items; // надо чтобы все итемы реализовывали интерфейс IMyInterface
}


Вот, собственно и всё. Как писать объявление? Просто public List<IMyInterface> Items?.. или надо сделать что-то навроде проперти, где в set проверять поддурживается ли интерфейс?..
PM MAIL WWW Skype   Вверх
HalkaR
Дата 28.6.2007, 12:54 (ссылка) |    (голосов:1) Загрузка ... Загрузка ... Быстрая цитата Цитата


Пуфыстый назгул
****


Профиль
Группа: Экс. модератор
Сообщений: 2132
Регистрация: 8.12.2002
Где: В Москве

Репутация: 14
Всего: 42



Kefir, имхо public List<IMyInterface> Items - самый простой способ. Чем он плох?
PM MAIL   Вверх
Kefir
Дата 28.6.2007, 13:59 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


«Hakuna Matata»
***


Профиль
Группа: Комодератор
Сообщений: 1878
Регистрация: 25.1.2003
Где: Tampere, Suomi

Репутация: 2
Всего: 87



гм.... хорошо....
делаю 
Цитата(HalkaR @  28.6.2007,  12:54 Найти цитируемый пост)
List<IMyInterface>


тогда например есть код:
Код

class MyClass : IMyInterface { ... }

Foo foo = new Foo();
List<MyClass> list = new List<MyClass>();
foo.Items = list;

тогда при компайле будет сообщение о том, что, мол, cannot implicitly convert type List<MyClass> to List<IMyInterface>. ведь MyClass имплементирует IMyInterface, тогда почему нельзя так? Или надо что-то где-то дописать? Или вообще положено писать так:
Код

List<IMyInterface> list = new List<IMyInterface>();
foo.Items = list;

? Но ведь тогда читабельность кода страдает (не видно какой на самом деле класс в листе)...
PM MAIL WWW Skype   Вверх
Void
Дата 28.6.2007, 14:12 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


λcat.lolcat
****


Профиль
Группа: Участник Клуба
Сообщений: 2206
Регистрация: 16.11.2004
Где: Zürich

Репутация: 25
Всего: 173



Kefir, то, что ты хочешь, называется ковариантность коллекций и достичь этого в шарпе, увы, невозможно.
Цитата(Kefir @  28.6.2007,  15:59 Найти цитируемый пост)
Но ведь тогда читабельность кода страдает (не видно какой на самом деле класс в листе)... 

Раз ты завёл интерфейс, значит в этом месте нас не должно волновать, что под ним прячется конкретно.

P.S. При том, что массивы-то ковариантны:
Код
class MyClass : IMyInterface { }
...
IMyInterface[] a = new MyClass[10]; // perfectly OK

Вот такой нелогичный... не язык даже, а рантайм.

Это сообщение отредактировал(а) Void - 28.6.2007, 14:15


--------------------
“Coming back to where you started is not the same as never leaving.” — Terry Pratchett
PM MAIL WWW GTalk   Вверх
Kefir
Дата 28.6.2007, 14:36 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


«Hakuna Matata»
***


Профиль
Группа: Комодератор
Сообщений: 1878
Регистрация: 25.1.2003
Где: Tampere, Suomi

Репутация: 2
Всего: 87



Цитата(Void @  28.6.2007,  14:12 Найти цитируемый пост)
ковариантность коллекций

 smile ээээ... мой уши в первый раз слышат это... ну ничего. теперь хоть знаю.


Цитата(Void @  28.6.2007,  14:12 Найти цитируемый пост)
Раз ты завёл интерфейс, значит в этом месте нас не должно волновать

гм.... ну я бы не сказал.
в общем суть проблемы от начала до конца.
у меня есть различные классы, которые будут реализовывать ICatalogueItem. есть юзер контрол, который показывает List<ICatalogueItem>. То есть примерно:
Код

class Catalogue : UserControl
{
  List<ICatalogueItem> Items;
}

interface ICatalogueItem
{
  string CatalogueText { get; }
}

class Drink : ICatalogueItem {
  public string Name;
  public string Vol;
  // etc.  

  public string CatalogueText { get { return this.Name; } }
}

class Food : ICatalogueItem {
  public string Name;
  public int Callories;
  //etc.

  public string CatalogueText { get { return this.Name; } }
}

так вот. в итоге где-то я буду получать (например после общения с БД) что-нибудь а-ля:
Код

List<Drink> drinks = Menu.GetAllTheDrinks();
catalogue1.Items = drinks;

// ...

List<Food> foods = Menu.GetAllTheFood();
catalogue1.Items = foods;

и чтоже теперь GetAllTheDrinks(); и GetAllTheFood(); должны будут возвращать List<ICatalogueItem> вместо логичных List<Drink> и List<Food>?..

Добавлено через 7 минут и 50 секунд
Цитата(Void @  28.6.2007,  14:12 Найти цитируемый пост)
Раз ты завёл интерфейс, значит в этом месте нас не должно волновать, что под ним прячется конкретно.

забыл - меня не должно волновать в самом юзер контроле Catalogue (в данном случае) какой класс мне дают на Items - главное чтобы класс реализовывал ICatalogueItem. Это же имхо для того и сделано, чтобы там не волноваться. А снаружи-то этого класса по сути Items назначаются так как хочешь (хоть List<Tralala> или List<Ugabuga> - главное чтобы Tralala : ICatalogueItem и Ugabuga : ICatalogueItem). Вот как в том же DataGridView - дата сорц может быть любой класс, реализовавший IList (ну или что-то вроде IList). Ну так я же беру и подаю ему DataTable или DataView, а не IList. Или я снова чего-то не понял? Почему у меня в первом после и упоминается object - потому что DataSource тоже object smile
PM MAIL WWW Skype   Вверх
mihryak
Дата 29.6.2007, 23:34 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Опытный
**


Профиль
Группа: Участник
Сообщений: 731
Регистрация: 28.4.2007
Где: С-Пб

Репутация: 19
Всего: 36



можно выкрутиться так (используя ковариантность массивов)
Код

            List<Drink> drinks = Menu.GetAllTheDrinks();
            catalogue1.Items.AddRange(drinks.ToArray());

            // ...

            List<Food> foods = Menu.GetAllTheFood();
            catalogue1.Items.AddRange(foods.ToArray());

рефлектором не смотрел, но вроде как серьёзных преобразований при этом не должно происходить

ПС. Не забываем выставлять заранее размер коллекции в catalogue1
Код

            catalogue1.Items = new List<ICatalogueItem>(drinks.Count);

PM MAIL ICQ   Вверх
Kefir
Дата 30.6.2007, 00:17 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


«Hakuna Matata»
***


Профиль
Группа: Комодератор
Сообщений: 1878
Регистрация: 25.1.2003
Где: Tampere, Suomi

Репутация: 2
Всего: 87



Цитата(mihryak @  29.6.2007,  23:34 Найти цитируемый пост)
catalogue1.Items.AddRange(foods.ToArray());

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

            catalogue1.Items = new List<ICatalogueItem>(categories.ToArray());

изменил только потому, что при AddRange надо сперва сделать Clear. но это мелочи, главное то, что setter у Items не происходит, где мне надо произвести некоторые обновления.

Добавлено через 2 минуты и 11 секунд
 smile 
хехе. ввёл в гугль строку "ковариантность коллекций" и эта тема - первый результат ))
гугль
PM MAIL WWW Skype   Вверх
Gelis
Дата 30.6.2007, 12:47 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Шустрый
*


Профиль
Группа: Участник
Сообщений: 145
Регистрация: 26.10.2006
Где: Брест

Репутация: 2
Всего: 4



Код

class Foo<T> where T : IMyInterface
{
    public List<T> Items;
}

PM MAIL   Вверх
anthony
Дата 30.6.2007, 13:34 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Новичок



Профиль
Группа: Участник
Сообщений: 19
Регистрация: 25.6.2007
Где: Украина

Репутация: 1
Всего: 5



Кстати третья ссылка в Гугль достаточно ясно описывает эту проблему в Java:

ковариантность коллекций
PM MAIL ICQ   Вверх
Idsa
Дата 6.7.2007, 20:23 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
****


Профиль
Группа: Участник
Сообщений: 2086
Регистрация: 5.12.2006
Где: Томск

Репутация: 14
Всего: 62



Я вот никак не могу разобраться с выводом по первому примеру в указанной anthony ссылке:
Цитата

List<Integer> li = new ArrayList<Integer>();
List<Number> ln = li; // не верно 
ln.add(new Float(3.1415));

Поскольку ln имеет тип List<Number>, добавление Float к нему выглядит совершенно легальным. Но если ln присвоить li, то это нарушило бы независимость от типа, который явно указан при определении li (что li - это список целых чисел), вот почему родовые (generic) типы не могут быть ковариантными.

В чем смысл?


--------------------
Мой блог: alexidsa.blogspot.com
PM MAIL ICQ   Вверх
Void
Дата 6.7.2007, 20:39 (ссылка) |    (голосов:1) Загрузка ... Загрузка ... Быстрая цитата Цитата


λcat.lolcat
****


Профиль
Группа: Участник Клуба
Сообщений: 2206
Регистрация: 16.11.2004
Где: Zürich

Репутация: 25
Всего: 173



List<T> — ссылочный тип. Если Integer — подтип Number и List<Integer> — подтип List<Number>, то должно быть возможно неявное приведение одного к другому. Таким образом, если ln присвоить li, то объявленный тип ln будет List<Number>, а фактический — List<Integer>. Когда мы добавим в ln объект типа Float, окажется, что li содержит и Integers, и Floats. И если самой коллекции может быть и всё равно, то код, работающий с ней, слегка удивиться, получив вещественное число там, где должны быть только целые.

Надо посмотреть, как это разрулено в Scala. Я только помню, что там ковариантность и контравариантность обобщённых типов указывается явно.


--------------------
“Coming back to where you started is not the same as never leaving.” — Terry Pratchett
PM MAIL WWW GTalk   Вверх
Idsa
Дата 7.7.2007, 21:39 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
****


Профиль
Группа: Участник
Сообщений: 2086
Регистрация: 5.12.2006
Где: Томск

Репутация: 14
Всего: 62



Цитата(Void @  6.7.2007,  20:39 Найти цитируемый пост)
Когда мы добавим в ln объект типа Float, окажется, что li содержит и Integers, и Floats.

Видимо, "окажется, что ln"?

Цитата(Void @  6.7.2007,  20:39 Найти цитируемый пост)
 И если самой коллекции может быть и всё равно, то код, работающий с ней, слегка удивиться, получив вещественное число там, где должны быть только целые.

А чего это он удивится? Тип-то Number.


--------------------
Мой блог: alexidsa.blogspot.com
PM MAIL ICQ   Вверх
Void
Дата 7.7.2007, 22:44 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


λcat.lolcat
****


Профиль
Группа: Участник Клуба
Сообщений: 2206
Регистрация: 16.11.2004
Где: Zürich

Репутация: 25
Всего: 173



Idsa, ln и li ссылаются на один и тот же объект, фактический тип которого — List<Integer>.

Кстати, глянул на коллекции в Scala. Вопрос решается вполне логично: ковариантны только неизменяемые коллекции.


--------------------
“Coming back to where you started is not the same as never leaving.” — Terry Pratchett
PM MAIL WWW GTalk   Вверх
  
Ответ в темуСоздание новой темы Создание опроса
Прежде чем создать тему, посмотрите сюда:
mr.DUDA
THandle

Используйте теги [code=csharp][/code] для подсветки кода. Используйтe чекбокс "транслит" если у Вас нет русских шрифтов.
Что делать если Вам помогли, но отблагодарить помощника плюсом в репутацию Вы не можете(не хватает сообщений)? Пишите сюда, или отправляйте репорт. Поставим :)
Так же не забывайте отмечать свой вопрос решенным, если он таковым является :)


Если Вам понравилась атмосфера форума, заходите к нам чаще! С уважением, mr.DUDA, THandle.

 
0 Пользователей читают эту тему (0 Гостей и 0 Скрытых Пользователей)
0 Пользователей:
« Предыдущая тема | Общие вопросы по .NET и C# | Следующая тема »


 




[ Время генерации скрипта: 0.0922 ]   [ Использовано запросов: 22 ]   [ GZIP включён ]


Реклама на сайте     Информационное спонсорство

 
По вопросам размещения рекламы пишите на vladimir(sobaka)vingrad.ru
Отказ от ответственности     Powered by Invision Power Board(R) 1.3 © 2003  IPS, Inc.