12.1.委托概述
12.1.2 委托的数据类型
class DelegateSample
{
static void Main(string[] args)
{
//int[] arr = { 10, 20, 30, 40, 50 };
int[] arr = { 50, 40, 30, 20, 10 };
ConsoleArr(arr);
ComparisonHandler wx = new ComparisonHandler(DelegateSample.IsTrue);
BubbleSort(arr, wx);
- //C#2.0之前是这么写的
- //BubbleSort(arr, new ComparisonHandler(IsTrue));
ConsoleArr(arr);
Console.Read();
}
public delegate bool ComparisonHandler(int a, int b);
public static bool IsTrue(int a, int b)
{
return a > b;
}
public static void BubbleSort(int[] items, ComparisonHandler comparisonMethod)
{
int i;
int j;
int temp;
if (items == null)
{
return;
}
if (comparisonMethod == null)
{
throw new ArgumentNullException("comparisonMethod");
}
for (i = items.Length - 1; i >= 0; i--)
{
for (j = 1; j <= i; j++)
{
if (comparisonMethod(items[j - 1], items[j]))
{
temp = items[j - 1];
items[j - 1] = items[j];
items[j] = temp;
}
}
}
}
public static void ConsoleArr(int[] arr)
{
foreach (var item in arr)
{
Console.Write(item+",");
}
Console.WriteLine();
}
}
public static bool AlphabeticalIsTrue(int a,int b)
{
int comparison;
comparison = (a.ToString().CompareTo(b.ToString()));
return comparison > 0;
}
//C# 2.0以后直接传递方法
BubbleSort(arr, AlphabeticalIsTrue);
12.1.3 委托内部机制
// 摘要:
// 初始化一个委托,该委托对指定的类实例调用指定的实例方法。
//
// 参数:
// target:
// 类实例,委托对其调用 method。
//
// method:
// 委托表示的实例方法的名称。
//
// 异常:
// System.ArgumentNullException:
// target 为 null。 - 或 - method 为 null。
//
// System.ArgumentException:
// 绑定到目标方法时出错。
[SecuritySafeCritical]
protected Delegate(object target, string method);
//
// 摘要:
// 初始化一个委托,该委托从指定的类调用指定的静态方法。
//
// 参数:
// target:
// System.Type,它表示定义 method 的类。
//
// method:
// 委托表示的静态方法的名称。
//
// 异常:
// System.ArgumentNullException:
// target 为 null。 - 或 - method 为 null。
//
// System.ArgumentException:
// target 不是 RuntimeType。 请参见 反射中的运行时类型。 - 或 - target 表示开放式泛型类型。
[SecuritySafeCritical]
protected Delegate(Type target, string method);
12.2.匿名方法
class Program
{
public delegate bool ComparisonHandler(int a, int b);
static void Main(string[] args)
{
int i;
int[] items = new int[5];
ComparisonHandler comparionMethod;
for (i = 0; i < items.Length; i++)
{
Console.WriteLine("Enter an integer:");
items[i] = int.Parse(Console.ReadLine());
}
comparionMethod = delegate(int first, int second)
{
return first < second;
};
BubbleSort(items, comparionMethod);
for ( i = 0; i < items.Length; i++)
{
Console.WriteLine(items[i]);
}
Console.Read();
}
public static void BubbleSort(int[] items, ComparisonHandler comparisonMethod)
{
int i;
int j;
int temp;
if (items == null)
{
return;
}
if (comparisonMethod == null)
{
throw new ArgumentNullException("comparisonMethod");
}
for (i = items.Length - 1; i >= 0; i--)
{
for (j = 1; j <= i; j++)
{
if (comparisonMethod(items[j - 1], items[j]))
{
temp = items[j - 1];
items[j - 1] = items[j];
items[j] = temp;
}
}
}
}
}
12.3.系统定义的委托:Func 和 Action 声明
公共语言运行时提供 Invoke 每种委托类型,具有相同的签名与委托的方法。 您不需要显式调用此方法,从 C#、 Visual Basic 或 Visual c + +,因为编译器会自动调用。 Invoke 方法就很有用 反射 如果想要查找的委托类型签名。
https://msdn.microsoft.com/zh-cn/library/system.delegate.aspx
12.4.语句Lambda
1.无参数的语句
static void Main(string[] args)
{
//...
Func<string> getUserInput =
() =>
{
string input;
do
{
input = Console.ReadLine();
}
while (input.Trim().Length == 0);
return input;
};
//...
}
2.只有一个参数的语句
IEnumerable<Process> processes = Process.GetProcesses()
.Where(process => { return process.WorkingSet64 > 100000000; });
12.5.表达式Lambda
BubbleSort(items, (first, second) => first > second);
BubbleSort(items, (first, second) =>
{
return first > second;
}
);
12.6.表达式树
1.Lambda表达式作为数据使用
class Program
{
static void Main(string[] args)
{
Person[] persons = new Person[3];
IEnumerable<Person> obj = persons.Where(person => person.Name.ToUpper() == "INIGO MONTOYA");
}
}
public class Person
{
public string Name { get; set; }
}
一个技术是将几百万数据传输到客户端,为每一行创建一个Person对象,根据Lambda创建一个委托,再针对每个Person执行这个委托。概念上和数组的情况一致,但代价过于昂贵。
.第二个技术要好很多,将Lambda的含义(过滤掉姓名不是INIGO MONTOYA的每一行)发给服务器。服务器将符合条件的少数几行传输到客户端;而不是先创建几百万个Person对象,再丢弃。但是没在那将Lambda的含义发送给服务器?
这正式语言中添加表达式树这一概念的动机。
对Person的Name属性的调用;
对string的ToUpper()方法调用;
一个常量值“INIGO MONTOYA”;
一个相等性操作符==。
2.表达式树作为对象图使用
3.委托和表达式树的比较
IEnumerable<Person> obj = persons.Where(person => person.Name.ToUpper() == "INIGO MONTOYA");
public static IEnumerable<TSource> Where<TSource>(
this IEnumerable<TSource> source,
Func<TSource, bool> predicate);
public static IQueryable<TSource> Where<TSource>(
this IQueryable<TSource> source,
Expression<Func<TSource, bool>> predicate)
4.解析表达式
(x,y)=>x>y
Func<int,int,bool>
名称 | 说明 | |
---|---|---|
Body | 获取 lambda 表达式的主体。 (继承自 LambdaExpression。) | |
CanReduce | 指示可将节点简化为更简单的节点。 如果返回 true,则可以调用 Reduce() 以生成简化形式。 (继承自Expression。) | |
Name | 获取 lambda 表达式的名称。 (继承自 LambdaExpression。) | |
NodeType | 返回此 Expression 的节点类型。 (继承自 LambdaExpression。) 在 Silverlight for Windows Phone Windows Phone OS 7.1 中,此成员是从 Expression.NodeType 中继承的。 在 XNA Framework Windows Phone OS 7.0 中,此成员是从 Expression.NodeType 中继承的。 | |
Parameters | 获取 lambda 表达式的参数。 (继承自 LambdaExpression。) | |
ReturnType | 获取 lambda 表达式的返回类型。 (继承自 LambdaExpression。) | |
TailCall | 获取一个值,该值指示是否将通过尾调用优化来编译 lambda 表达式。 (继承自 LambdaExpression。) | |
Type | 获取此 Expression 所表示的表达式的静态类型。 (继承自 LambdaExpression。) 在 Silverlight for Windows Phone Windows Phone OS 7.1 中,此成员是从 Expression.Type 中继承的。 在 XNA Framework Windows Phone OS 7.0 中,此成员是从 Expression.Type 中继承的。 |
Expression<Func<int, int, bool>> expression;
expression = (x, y) => x > y;
Console.WriteLine("------{0}------",expression);
PrintNode(expression.Body, 0);//expression.Body: (x > y) [lambda 表达式的主体]
Console.WriteLine();
Console.WriteLine();
expression = (x, y) => x * y > x + y;
Console.WriteLine("------{0}------",expression);
PrintNode(expression.Body, 0); //expression.Body: (x * y) > (x + y)
/*
* expression.Body: (x * y) > (x + y) [lambda 表达式的主体]
* BinaryExpression是否具有二进制运算符:true
* expression.Left:(x * y)
* BinaryExpression是否具有二进制运算符:true
* expression.Left:x expression.NodeType:Parameter
* BinaryExpression是否具有二进制运算符:false
* expression.Left:* expression.NodeType:Multiply
* .....
*/
Console.WriteLine();
Console.WriteLine();
Console.Read();
}
public static void PrintNode(Expression expression, int indent)
{
if (expression is BinaryExpression) //具有二进制运算符的表达式
PrintNode(expression as BinaryExpression, indent);
else
PrintSingle(expression, indent);
}
private static void PrintNode(BinaryExpression expression, int indent)
{
PrintNode(expression.Left, indent + 1);
PrintSingle(expression, indent);
PrintNode(expression.Right, indent + 1);
}
private static void PrintSingle(Expression expression, int indent)
{
Console.WriteLine("{0," + indent * 5 + "}{1}", "", NodeToSting(expression));
}
private static string NodeToSting(Expression expression)
{
switch (expression.NodeType)
{
case ExpressionType.Multiply:
return "*";
case ExpressionType.Add:
return "+";
case ExpressionType.Divide:
return "/";
case ExpressionType.Subtract:
return "-";
case ExpressionType.GreaterThan:
return ">";
case ExpressionType.LessThan:
return "<";
default:
return expression.ToString() + "(" + expression.NodeType.ToString() + ")";
}
}
注意:Lambda语句不能转换成表达式树,只有表达式Lambda才能转为表达式树。