В этой версии программы ограничение на интерфейс, указываемое в классе PhoneList
, требует, чтобы аргумент типа реализовал интерфейс IPhoneList
. А поскольку этот интерфейс реализуется в обоих классах, Friend
и Supplier
, то они относятся к допустимым типам, привязываемым к типу Т. В то же время интерфейс не реализуется в классе EmailFriend
, и поэтому этот класс не может быть привязан к типу Т. Для того чтобы убедиться в этом, удалите символы комментария в двух последних строках кода в методе Main(). Вы сразу же обнаружите, что программа не компилируется.
Ограничение new()
на конструктор позволяет получать экземпляр объекта обобщенного типа. Как правило, создать экземпляр параметра обобщенного типа не удается. Но это положение изменяет ограничение new()
, поскольку оно требует, чтобы аргумент типа предоставил конструктор без параметров. Им может быть конструктор, вызываемый по умолчанию и предоставляемый автоматически, если явно определяемый конструктор отсутствует или же конструктор без параметров явно объявлен пользователем. Накладывая ограничение new()
, можно вызывать конструктор без параметров для создания объекта.
Ниже приведен простой пример, демонстрирующий наложение ограничения new()
.
// Продемонстрировать наложение ограничения new() на конструктор.
using System;
class MyClass {
public MyClass() {
// ...
}
//. . .
}
class Test
T obj;
public Test() {
// Этот код работоспособен благодаря
// наложению ограничения new().
obj = new T(); // создать объект типа Т
}
// ...
}
class ConsConstraintDemo {
static void Main() {
Test
}
}
Прежде всего обратите внимание на объявление класса Test.
class Test
В силу накладываемого ограничения new()
любой аргумент типа должен предоставлять конструктор без параметров.
Далее проанализируем приведенный ниже конструктор класса Test.
public Test() {
// Этот код работоспособен благодаря
// наложению ограничения new().
obj = new Т(); // создать объект типа Т
}
В этом фрагменте кода создается объект типа Т, и ссылка на него присваивается переменной экземпляра obj
. Такой код допустим только потому, что ограничение new()
требует наличия конструктора. Для того чтобы убедиться в этом, попробуйте сначала удалить ограничение new()
, а затем попытайтесь перекомпилировать программу. В итоге вы получите сообщение об ошибке во время компиляции.
В методе Main() получается экземпляр объекта типа Test, как показано ниже.
Test
Обратите внимание на то, что аргументом типа в данном случае является класс MyClass
и что в этом классе определяется конструктор без параметров. Следовательно, этот класс допускается использовать в качестве аргумента типа для класса Test. Следует особо подчеркнуть, что в классе MyClass
совсем не обязательно определять конструктор без параметров явным образом. Его используемый по умолчанию конструктор вполне удовлетворяет накладываемому ограничению. Но если классу потребуются другие конструкторы, помимо конструктора без параметров, то придется объявить явным образом и вариант без параметров.
Что касается применения ограничения new()
, то следует обратить внимание на три других важных момента. Во-первых, его можно использовать вместе с другими ограничениями, но последним по порядку. Во-вторых, ограничение new()
позволяет конструировать объект, используя только конструктор без параметров, — даже если доступны другие конструкторы. Иными словами, передавать аргументы конструктору параметра типа не разрешается. И в-третьих, ограничение new()
нельзя использовать одновременно с ограничением типа значения, рассматриваемым далее.
Два других ограничения позволяют указать на то, что аргумент, обозначающий тип, должен быть либо ссылочного типа, либо типа значения. Эти ограничения оказываются полезными в тех случаях, когда для обобщенного кода важно провести различие между ссылочным типом и типом значения. Ниже приведена общая форма ограничения ссылочного типа.
where Т : class
В этой форме с оператором where
ключевое слово class
указывает на то, что аргумент int
или bool
, вместо T приведет к ошибке во время компиляции.
Ниже приведена общая форма ограничения типа значения.
where Т : struct