【C#】委托
1.委托 delegate
委托是一种引用类型,能封装一个或者多个方法。类似于C++中的函数指针,但是比指针更灵活、安全。
(图片来源网络,侵删)
C++中函数指针通常用于:
1.动态调用函数:在编译时不调用,而是在运行时。
2.回调机制:异步操作中
3.实现策略模式:允许在运行时选择算法或策略
4.简化代码:将函数作为参数传入,封装好只用一个函数调用。
简单的委托代码:
using System;
namespace DelegateExample
{
// 定义一个委托类型,用于封装数学操作
public delegate double MathOperation(double a, double b);
class Program
{
static void Main(string[] args)
{
// 创建委托实例并指向不同的数学操作方法
// 两种写法,都可以
MathOperation add = new MathOperation(Add);
MathOperation subtract = new MathOperation(Subtract);
MathOperation multiply = Multiply;
MathOperation divide = Divide;
// 使用委托调用方法
double x = 10;
double y = 5;
Console.WriteLine($"Adding {x} and {y}: {add(x, y)}");
Console.WriteLine($"Subtracting {y} from {x}: {subtract(x, y)}");
Console.WriteLine($"Multiplying {x} and {y}: {multiply(x, y)}");
Console.WriteLine($"Dividing {x} by {y}: {divide(x, y)}");
Console.ReadKey();
}
// 加法
public static double Add(double a, double b)
{
return a + b;
}
// 减法
public static double Subtract(double a, double b)
{
return a - b;
}
// 乘法
public static double Multiply(double a, double b)
{
return a * b;
}
// 除法
public static double Divide(double a, double b)
{
if (b == 0)
{
throw new DivideByZeroException("Divider cannot be zero.");
}
return a / b;
}
}
}
上述代码可能感受不直观,下面是两组对比:
前者是正常的函数实现,可以发现代码重复量很多,方法结构、传参方式很相似。
using System;
using System.Collections.Generic;
class Program
{
static double Sum(List values)
{
double result = 0;
foreach (double value in values)
{
result += value;
}
return result;
}
static double Average(List values)
{
if (values.Count == 0) return 0;
return Sum(values) / values.Count;
}
static double Max(List values)
{
if (values.Count == 0) return 0;
double result = values[0];
foreach (double value in values)
{
if (value > result)
{
result = value;
}
}
return result;
}
static double Min(List values)
{
if (values.Count == 0) return 0;
double result = values[0];
foreach (double value in values)
{
if (value
下面是用函数委托来实现:
using System;
using System.Collections.Generic;
class Program
{
// 定义一个委托类型,用于封装数学操作
public delegate double MathOperation(double a, double b);
// 定义一个通用的操作方法
static double ApplyOperation(List values, MathOperation operation, double initialValue)
{
double result = initialValue;
foreach (double value in values)
{
result = operation(result, value);
}
return result;
}
// 各种具体的操作方法
static double SumOperation(double a, double b) => a + b;
static double MaxOperation(double a, double b) => (a > b) ? a : b;
static double MinOperation(double a, double b) => (a ApplyOperation(values, SumOperation, 0);
static double Average(List values) => values.Count == 0 ? 0 : Sum(values) / values.Count;
static double Max(List values) => values.Count == 0 ? 0 : ApplyOperation(values, MaxOperation, values[0]);
static double Min(List values) => values.Count == 0 ? 0 : ApplyOperation(values, MinOperation, values[0]);
static void Main()
{
List values = new List { 1.5, 2.5, 3.5, 4.5 };
Console.WriteLine("Sum: " + Sum(values));
Console.WriteLine("Average: " + Average(values));
Console.WriteLine("Max: " + Max(values));
Console.WriteLine("Min: " + Min(values));
}
}
2.Lambda表达式
本质为匿名函数,现代C#已经用lambda表达式逐步代替匿名函数了。
语法格式:
(参数列表)=> {语句序列};
因为lambda依附于委托,所以也要遵循委托的限制。
1.参数列表中的参数个数应该由相应的委托限制。
2.参数列表中限制只有一个参数时,参数列表()括号可以省略。
3.当编译器能自动推断参数类型时,参数列表中可以不写类型。
4.委托声明中对参数加入了ref和out限制,lambda中也要加上。
5.当右侧语句序列中只有一条语句,{}花括号可以省略。
6.有返回值,必须用return;如果语句序列中只有一个时,return可以省略。
7.如果委托有返回值类型,则lambda也必须有同类型返回值。
免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们,邮箱:ciyunidc@ciyunshuju.com。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!
