Следует особо подчеркнуть, что переменной AlphaRef
можно присвоить ссылку на объект типа MyClass
благодаря только тому, что обобщенный тип Т указан как ковариантный в интерфейсе IMyCoVarGenIF
. Для того чтобы убедиться в этом, удалите ключевое слово out
из объявления параметра обобщенного типа Т в интерфейсе IMyCoVarGenIF
и попытайтесь скомпилировать данную программу еще раз. Компиляция завершится неудачно, поскольку строгая проверка на соответствие типов не разрешит теперь подобное присваивание.
Один обобщенный интерфейс может вполне наследовать от другого. Иными словами, обобщенный интерфейс с параметром ковариантного типа можно расширить, как показано ниже.
public interface IMyCoVarGenIF2
// ...
}
Обратите внимание на то, что ключевое слово out
указано только в объявлении расширенного интерфейса. Указывать его в объявлении базового интерфейса не только не нужно, но и не допустимо. И последнее замечание: обобщенный тип Т допускается не указывать как ковариантный в объявлении интерфейса IMyCoVarGenIF2
. Но при этом исключается ковариантность, которую может обеспечить расширенный интерфейс IMyCoVarGetlF
. Разумеется, возможность сделать интерфейс IMyCoVarGenIF2
инвариантным может потребоваться в некоторых случаях его применения.
На применение ковариантности накладываются некоторые ограничения. Ковариантность параметра типа может распространяться только на тип, возвращаемый методом. Следовательно, ключевое слово out нельзя применять в параметре типа, служащем для объявления параметра метода. Ковариантность оказывается пригодной только для ссылочных типов. Ковариантный тип нельзя использовать в качестве ограничения в интерфейсном методе. Так, следующий интерфейс считается недопустимым.
public interface IMyCoVarGenIF2
void M
// использовать как ограничение
}
Применительно к обобщенному интерфейсу контравариантность служит средством, разрешающим методу использовать аргумент, тип которого относится к базовому классу, указанному в соответствующем параметре типа. В прошлом тип аргумента метода должен был в точности соответствовать параметру типа в силу строгой проверки обобщений на соответствие типов. Контравариантность смягчает это строгое правило таким образом, чтобы обеспечить типовую безопасность. Параметр контравариантного типа объявляется с помощью ключевого слова in
, которое предваряет имя этого параметра.
Для того чтобы стали понятнее последствия применения ковариантности, вновь обратимся к конкретному примеру. Ниже приведен обобщенный интерфейс IMyContraVarGenIF
контравариантного типа. В нем указывается контравариантный параметр обобщенного типа Т, который используется в объявлении метода Show()
.
// Это обобщенный интерфейс, поддерживающий контравариантность.
public interface IMyContraVarGenIF
void Show(T obj);
}
Как видите, обобщенный тип Т указывается в данном интерфейсе как контравариантный с помощью ключевого слова in
, предшествующего имени его параметра. Обратите также внимание на то, что Т является параметром типа для аргумента obj
в методе Show()
.
Далее интерфейс IMyContraVarGenIF реализуется в классе MyClass, как показано ниже.
// Реализовать интерфейс IMyContraVarGenIF.
class MyClass
public void Show(T x) {
Console.WriteLine(x);
}
}
В данном случае метод Show()
просто выводит на экран строковое представление переменной х, получаемое в результате неявного обращения к методу ToString()
из метода WriteLine()
.
После этого объявляется иерархия классов, как показано ниже.
// Создать простую иерархию классов,
class Alpha {
public override string ToString() {
return "Это объект класса Alpha.";
}
// ...
}
class Beta : Alpha {
public override string ToString() {
return "Это объект класса Beta.";
}
// ...
}
Ради большей наглядности классы Alpha
и Beta
несколько отличаются от аналогичных классов из предыдущего примера применения ковариантности. Обратите также внимание на то, что метод ToString()
переопределяется таким образом, чтобы возвращать тип объекта.
С учетом всего изложенного выше, следующая последовательность операций будет считаться вполне допустимой.