C# 4.0 полное руководство - 2011
Шрифт:
Func<int, int, bool> IsFactor = IsFactorExp.Compile;
После выполнения этого оператора делегат IsFactorExp может быть вызван, чтобы определить, является ли одно целое число множителем другого.
Обратите также внимание на то, что <Func<int, int, boo 1> обозначает тип делегата. В этой форме делегата Fun с указываются два параметра типа int и возвращаемый тип bool. В рассматриваемой здесь программе использована именно эта форма делегата Fun с, совместимая
Методы расширения
Как упоминалось выше, методы расширения предоставляют средства для расширения функций класса, не прибегая к обычному механизму наследования. Методы расширения создаются нечасто, поскольку механизм наследования, как правило, предлагает лучшее решение. Тем не менее знать, как они действуют, никогда не помешает. Ведь они имеют существенное значение для LINQ.
Метод расширения является статическим и поэтому должен быть включен в состав статического, необобщенного класса. Тип первого параметра метода расширения определяет тип объектов, для которых этот метод может быть вызван. Кроме того, первый параметр может быть указан с модификатором this. Объект, для которого вызывается метод расширения, автоматически передается его первому параметру. Он не передается явным образом в списке аргументов. Следует, однако, иметь в виду, что метод расширения может по-прежнему вызываться для объекта аналогично методу экземпляра, несмотря на то, что он объявляется как статический.
Ниже приведена общая форма метода расширения.
static возращаемый_тип имя (this тип_вызывающего_объекта ob, список_параметров)
Очевидно, что список_параметров окажется пустым в отсутствие аргументов, за исключением аргумента, неявно передаваемого вызывающим объектом оЬ. Не следует, однако, забывать, что первым параметром метода расширения является автоматически передаваемый объект, для которого вызывается этот метод. Как правило, метод расширения становится открытым членом своего класса.
В приведенном ниже примере программы создаются три простых метода расширения.
// Создать и использовать ряд методов расширения, using System;
using System.Globalization; static class MyExtMeths {
// Возвратить обратную величину числового значения типа double, public static double Reciprocal(this double v) { return 1.0 / v;
}
// Изменить на обратный регистр букв в символьной // строке и возвратить результат, public static string RevCase(this string str) { string temp =
foreach(char ch in str) {
if(Char.IsLower(ch)) temp += Char.ToUpper (ch, Culturelnfo.
CurrentCulture);
else temp += Char.ToLower(ch, Culturelnfo.CurrentCulture);
}
return temp;
}
// Возвратить абсолютное значение выражения n / d. public static double AbsDivideBy(this double n, double d) { return Math.Abs(n / d);
}
}
class ExtDemo {
static void Main {
double val = 8.0;
string str = "Alpha Beta Gamma";
// Вызвать метод расширения Reciprocal..
Console.WriteLine("Обратная величина {0} равна {1}", val, val.Reciprocal);
// Вызвать метод расширения RevCaseO .
Console.WriteLine(str + " после смены регистра: " + str.RevCase );
// Использовать метод расширения AbsDivideBy .
Console.WriteLine("Результат вызова метода val.AbsDivideBy(-2) : " + val.AbsDivideBy(-2));
}
}
Обратная величина 8 равна 0.125
Alpha Beta Gamma после смены регистра: aLPHA ЬЕТА дАММА Результат вызова метода val.AbsDivideBy(-2): 4
В данном примере программы каждый метод расширения содержится в статическом классе MyExtMeths. Как пояснялось выше, метод расширения должен быть объявлен в статическом классе. Более того, этот класс должен находиться в области действия своих методов расширения, чтобы ими можно было пользоваться. (Именно поэтому в исходный текст программы следует включить пространство имен System. Linq, так как это дает возможность пользоваться методами расширения, связанными с LINQ.)
Объявленные методы расширения вызываются для объекта таким же образом, как и методы экземпляра. Главное отличие заключается в том, что вызывающий объект передается первому параметру метода расширения. Поэтому при выполнении выражения
val.AbsDivideBy(-2)
объект val передается параметру п метода расширения AbsDivideBy , а значение – 2 — параметру d.
Любопытно, что методы расширения Reciprocal и AbsDivideBy могут вполне законно вызываться и для литерала типа double, как показало ниже, поскольку они определены для этого типа данных.
8.0.Reciprocal
8.0.AbsDivideBy(-1)
Кроме того, метод расширения RevCase может быть вызван следующим образом. "AbCDe".RevCase
В данном случае возвращается строковый литерал с измененным на обратный регистром букв.
PLINQ
В версии .NET Framework 4.0 внедрено новое дополнение LINQ под названием PLINQ. Это средство предназначено для поддержки параллельного программирования. Оно позволяет автоматически задействовать в запросе несколько доступных процессоров. Подробнее о PLINQ и других средствах, связанных с параллельным програмт-мированием, речь пойдет в главе 24.
ГЛАВА 20 Небезопасный код, указатели, обнуляемые типы и разные ключевые слова
В этой главе рассматривается средство языка С#, которое обычно захватывает программистов врасплох. Это небезопасный код. В таком коде зачастую используются указатели. Совместно с небезопасным кодом указатели позволяют разрабатывать на C# приложения, которые обычно связываются с языком C++, высокой производительностью и системным кодом. Более того, благодаря включению небезопасного кода и указателей в состав C# в этом языке появились возможности, которые отсутствуют в Java.