T[] src, T[] target) {
Параметр типа объявляется CopyInsert()
является статическим, что позволяет вызывать его независимо от любого объекта. Следует, однако, иметь в виду, что обобщенные методы могут быть либо статическими, либо нестатическими. В этом отношении для них не существует никаких ограничений.
Далее обратите внимание на то, что метод CopyInsert()
вызывается в методе Main()
с помощью обычного синтаксиса и без указания аргументов типа. Дело в том, что типы аргументов различаются автоматически, а тип Т соответственно подстраивается. Этот процесс называется
ArrayUtils.CopyInsert (99, 2, nums, nums2);
тип T становится типом int
, поскольку числовое значение 99 и элементы массивов nums и nums2 относятся к типу int
. А во втором вызове данного метода используются строковые типы, и поэтому тип Т заменяется типом string
.
А теперь обратите внимание на приведенную ниже закомментированную строку кода.
// ArrayUtils.CopyInsert(0.01, 2, nums, nums2);
Если удалить символы комментария в начале этой строки кода и затем попытаться перекомпилировать программу, то будет получено сообщение об ошибке. Дело в том, что первый аргумент в данном вызове метода CopyInsert()
относится к типу double
, а третий и четвертый аргументы обозначают элементы массивов nums и nums2 типа int
. Но все эти аргументы типа должны заменить один и тот же параметр типа Т, а это приведет к несоответствию типов и, как следствие, к ошибке во время компиляции. Подобная возможность соблюдать типовую безопасность относится к одним из самых главных преимуществ обобщенных методов.
Синтаксис объявления метода CopyInsert()
может быть обобщен. Ниже приведена общая форма объявления обобщенного метода.
возвращаемый_ тип имя_метода<список_параметров_типа> (список_параметров) {
// ...
В любом случае
В большинстве случаев неявной выводимости типов оказывается достаточно для вызова обобщенного метода, тем не менее аргументы типа могут быть указаны явным образом. Для этого достаточно указать аргументы типа после имени метода при его вызове. В качестве примера ниже приведена строка кода, в которой метод CopyInsert()
вызывается с явно указываемым аргументом типа string.
ArrayUtils.CopyInsert
Тип передаваемых аргументов необходимо указывать явно в том случае, если компилятор не сможет вывести тип параметра Т или если требуется отменить выводимость типов.
На аргументы обобщенного метода можно наложить ограничения, указав их после списка параметров. В качестве примера ниже приведен вариант метода CopyInsert() для обработки данных только ссылочных типов.
public static bool CopyInsert
T[] src, T[] target) where T : class {
Если попробовать применить этот вариант в предыдущем примере программы обработки массивов, то приведенный ниже вызов метода CopyInsert ()
не будет скомпилирован, поскольку int является типом значения, а не ссылочным типом.
// Теперь неправильно, поскольку параметр Т должен быть ссылочного типа!
ArrayUtils.CopyInsert(99, 2, nums, nums2); // Теперь недопустимо!
Обобщенные делегаты
Как и методы, делегаты также могут быть обобщенными. Ниже приведена общая форма объявления обобщенного делегата.
delegate возвратцаемый_тип имя_делегата<список_параметров_типа> (список_аргументов) ;
Обратите внимание на расположение списка параметров типа. Он следует непосредственно после имени делегата. Преимущество обобщенных делегатов заключается в том, что их допускается определять в типизированной обобщенной форме, которую можно затем согласовать с любым совместимым методом.
В приведенном ниже примере программы демонстрируется применение делегата SomeOp
с одним параметром типа Т. Этот делегат возвращает значение типа Т и принимает аргумент типа Т.
// Простой пример обобщенного делегата,
using System;
// Объявить обобщенный делегат,