Шилдт Герберт
Шрифт:
• Ограничение на интерфейс, требующее реализации одного или нескольких интерфейсов аргументом типа. Это ограничение накладывается указанием имени требуемого интерфейса.
• Ограничение на конструктор, требующее предоставить конструктор без параметров в аргументе типа. Это ограничение накладывается с помощью оператора new .
• Ограничение ссылочного типа, требующее указывать аргумент ссылочного типа с помощью оператора class.
• Ограничение типа значения, требующее указывать аргумент типа значения с помощью оператора struct.
Среди всех этих ограничений чаще всего применяются ограничения на базовый класс и интерфейс, хотя все они важны в равной степени. Каждое из этих ограничений рассматривается далее по порядку.
Применение ограничения на базовый класс
Ограничение на базовый класс позволяет указывать базовый класс, который должен наследоваться аргументом типа. Ограничение на базовый класс служит двум главным целям. Во-первых, оно позволяет использовать в обобщенном классе те члены базового класса, на которые указывает данное ограничение. Это дает, например, возможность вызвать метод или обратиться к свойству базового класса. В отсутствие ограничения на базовый класс компилятору ничего не известно о типе членов, которые может иметь аргумент типа. Накладывая ограничение на базовый класс, вы тем самым даете компилятору знать, что все аргументы типа будут иметь члены, определенные в этом базовом классе.
И во-вторых, ограничение на базовый класс гарантирует использование только тех аргументов типа, которые поддерживают указанный базовый класс. Это означает, что для любого ограничения, накладываемого на базовый класс, аргумент типа должен обозначать сам базовый класс или производный от него класс. Если же попытаться использовать аргумент типа, не соответствующий указанному базовому классу или не наследующий его, то в результате возникнет ошибка во время компиляции. /
Ниже приведена общая форма наложения ограничения на базовый класс, в которой используется оператор where:
where Т : имя_базового_класса
где Г обозначает имя параметра типа, а имя_базового_класса — конкретное имя ограничиваемого базового класса. Одновременно в этой форме ограничения может быть указан только один базовый класс.
В приведенном ниже простом примере демонстрируется механизм наложения ограничения на базовый класс.
// Простой пример, демонстрирующий механизм наложения // ограничения на базовый класс.
using System;
class А {
public void Hello {
Console.WriteLine("Hello");
// Класс В наследует класс А. class В : А { }
// Класс С не наследует класс А. class С '{ }
//В силу ограничения на базовый класс во всех аргументах типа,
// передаваемых классу Test, должен присутствовать базовый класс А. class Test<T> where Т : А {
Т obj;
public Test(Т о) { obj = о;
}
public void SayHelloO {
// Метод Hello вызывается, поскольку он объявлен в базовом классе А. obj.Hello;
}
}
class BaseClassConstraintDemo { static void Main {
A a = new A;
В b = new В
С с = new Сj
// Следующий код вполне допустим, поскольку класс А указан как базовый.
Test<A> tl = new Test<A>(a);
tl.SayHello;
// Следующий код вполне допустим, поскольку класс В наследует от класса А.
Test<B> t2 = new Test<B> (b);
t2.SayHello;
// Следующий код недопустим, поскольку класс С не наследует от класса А.
// Test<C> t3 = new Test<C>(c); // Ошибка!
// t3.SayHello; // Ошибка!
}
}
В данном примере кода класс А наследуется классом В, но не наследуется классом С. Обратите также внимание на то, что в классе А объявляется метод Hello , а класс Test объявляется как обобщенный следующим образом.