}
}
Вот к какому результату приводит выполнение этой программы.
Число 5 является множителем 10.
Число 7 не является множителем 10.
Данный пример программы наглядно показывает два основных этапа применения дерева выражений. Сначала в ней создается дерево выражений с помощью следующего оператора.
Expression<Func<int, int, bool»
IsFactorExp = (n, d) => (d != 0) ? (n % d) ==0 : false;
В этом операторе конструируется представление лямбда-выражения в оперативной памяти. Как пояснялось выше, это представление доступно по ссылке, присваиваемой делегату IsFactorExp. А в следующем операторе данные выражения преобразуются в исполняемый код.
Func<int, int, bool> IsFactor = IsFactorExp.Compile();
После выполнения этого оператора делегат IsFactorExp может быть вызван, чтобы определить, является ли одно целое число множителем другого.
Обратите также внимание на то, что <Func<int, int, boo 1> обозначает тип делегата. В этой форме делегата Fun с указываются два параметра типа int и возвращаемый тип bool. В рассматриваемой здесь программе использована именно эта форма делегата Fun с, совместимая с лямбда-выражениями, поскольку для выражения требуются два параметра. Для других лямбда-выражений могут подойти иные формы делегата Fun с в зависимости от количества требуемых параметров. Вообще говоря, конкретная форма делегата Fun с должна удовлетворять требованиям лямбда-выражения.
Методы расширения
Как упоминалось выше, методы расширения предоставляют средства для расширения функций класса, не прибегая к обычному механизму наследования. Методы расширения создаются нечасто, поскольку механизм наследования, как правило, предлагает лучшее решение. Тем не менее знать, как они действуют, никогда не помешает. Ведь они имеют существенное значение для LINQ.
Метод расширения является статическим и поэтому должен быть включен в состав статического, необобщенного класса. Тип первого параметра метода расширения определяет тип объектов, для которых этот метод может быть вызван. Кроме того, первый параметр может быть указан с модификатором this. Объект, для которого вызывается метод расширения, автоматически передается его первому параметру. Он не передается явным образом в списке аргументов. Следует, однако, иметь в виду, что метод расширения может по-прежнему вызываться для объекта аналогично методу экземпляра, несмотря на то, что он объявляется как статический.
Ниже приведена общая форма метода расширения.
static
Очевидно, что
В приведенном ниже примере программы создаются три простых метода расширения.
// Создать и использовать ряд методов расширения, 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);
}
