C# 如何自定义LINQ扩展方法 - 为IEnumerable添加新功能
技术百科
幻夢星雲
发布时间:2025-12-25
浏览: 次 C#中为IEnumerable添加自定义LINQ扩展方法需满足三条件:定义在非泛型静态类中、方法为静态、首个参数用this修饰IEnumerable;例如GetOrDefault扩展可安全取第N个元素并越界返回默认值。
添加新功能">
在 C# 中,为 IEnumerable 添加自定义 LINQ 扩展方法,本质是写一个 静态类中的静态方法,且第一个参数用 this 修饰符绑定到 IEnumerable 类型。这样就能像原生 Where、Select 那样链式调用。
扩展方法必须满足的三个条件
缺一不可,否则编译器不认作扩展方法:
- 方法必须定义在非泛型静态类中(比如叫
EnumerableExtensions) - 方法本身必须是静态方法
- 第一个参数必须是
this IEnumerable(或具体类型如source this IEnumerable),且source T要在方法签名中声明为泛型参数
写一个实用的扩展:GetOrDefault
类似字典的 TryGetValue,但用于集合——安全取第 N 个元素,越界时返回默认值而非异常:
public static class EnumerableExtensions
{
public static T GetOrDefault(this IEnumerable source, int index, T defaultValue = default)
{
if (source == null) throw new ArgumentNullException(nameof(source));
if (index < 0) return defaultValue;
// 尽量避免 ToList(),用迭代器高效处理
var enumerator = source.GetEnumerator();
for (int i = 0; i <= index; i++)
{
if (!enumerator.MoveNext())
return defaultValue;
if (i == index)
return enumerator.Current;
}
return defaultValue;
}}
使用示例:
var list = new[] { "a", "b", "c" };
Console.WriteLine(list.GetOrDefault(1)); // "b"
Console.WriteLine(list.GetOrDefault(5)); // null(string 默认值)
Console.WriteLine(list.GetOrDefault(5, "N/A")); // "N/A"注意性能与空值安全
扩展方法不是魔法,它只是语法糖。实际调用仍走迭代逻辑,所以要注意:
- 避免在方法内部无脑调用
ToList()或ToArray(),尤其对大数据流或 IO 枚举器(如文件行枚举)会造成额外内存和延迟 - 始终校验
source是否为null,并给出清晰异常信息 - 如果扩展行为依赖索引(如分页、取第 N 项),考虑是否支持
IList优化路径(用source is IList)list ? list[index] : ...
让扩展方法支持链式调用和延迟执行
保持与标准 LINQ 一致的行为:返回 IEnumerable、不立即执行、支持 yield return:
public static IEnumerableWhereNotNull (this I Enumerable
source) where T : class { if (source == null) throw new ArgumentNullException(nameof(source)); foreach (T item in source) { if (item != null) yield return item; } }
这样就能无缝接入现有链式调用:
var result = items.Where(x => x.Length > 3)
.WhereNotNull()
.Select(x => x.ToUpper());基本上就这些。核心就是“静态类 + 静态方法 + this 参数”,再加一点对延迟执行和空值的敬畏。写多了你会发现,自己写的扩展和 LINQ 原生方法用起来几乎没区别。
# 就能
# 大数据
# 第一个
# 链式
# 分页
# 自定义
# 要注意
# 要在
# 迭代
# String
# 泛型
# 区别
# c#
# this
# NULL
# 类中
# select
# 默认值
# linq
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- SAX解析器是什么,它与DOM在处理大型XML文件
- Go语言中正确反序列化多个同级XML元素为结构体切
- Win11怎么更改输入法顺序_Win11调整语言首
- Win11怎么开启游戏模式_Win11优化游戏帧数
- Go 中 defer 在 goroutine 内部
- Win11怎么打开注册表_Windows 11注册
- Python安全爬虫设计_IP代理池与验证码识别策
- 如何在Golang中编写端到端测试_Golang
- 如何在 Go 中可靠地测试含 time.Time
- php订单日志怎么记录发货_php记录订单发货操作
- 如何开启Windows的远程服务器管理工具(RSA
- Win11怎么设置ipv4地址_Windows 1
- Win11怎么关闭自动维护 Win11禁用系统自动
- Win11怎么清理C盘虚拟内存_Win11清理虚拟
- php本地部署后session无法保存_sessi
- c++获取当前时间戳_c++ time函数使用详解
- 如何使用Golang进行HTTP服务性能测试_测量
- Win11怎么关闭应用权限_Windows11相机
- php嵌入式多设备通信怎么实现_php同时管理多个
- Windows系统时间服务错误_W32Time服务
- 如何在Golang中写入XML文件_生成符合规范的
- Linux怎么查找死循环进程_Linux系统负载分
- Win11怎么压缩文件 Win11自带压缩解压功能
- Win11如何设置环境变量 Win11添加和修改系
- Windows电脑如何进入安全模式?(多种按键方法
- Python与Docker容器化部署实战_镜像构建
- php串口通信波特率怎么选_根据硬件手册设置正确波
- Windows10电脑怎么设置虚拟光驱_Win10
- C++如何编写函数模板?(泛型编程入门)
- Win11怎么退出高对比度模式_Win11取消反色
- 如何在 Go 中正确反序列化多个同级 XML 元素
- Win11如何更新显卡驱动 Win11检查和安装设
- Win10如何卸载微软拼音输入法 Win10只保留
- Windows执行文件被SmartScreen拦截
- Windows Defender扫描失败怎么办_安
- php打包exe如何加密代码_防反编译保护方法【技
- Win11怎么查看显卡温度 Win11任务管理器查
- Win11怎么查看激活状态_查询Windows 1
- 如何在 Go 中高效缓存与分发网络视频流
- Win11怎么关闭搜索历史 Win11清除搜索框最
- 如何使用正则表达式提取以编号开头、后跟多个注解的完
- C#如何使用Channel C#通道实现异步通信
- PHP接收参数值为空怎么办_判断和处理空参数方法说
- c++怎么处理多线程死锁_c++ lock_gua
- c++协程和线程的区别 c++异步编程模型对比【核
- c++如何利用doxygen生成开发文档_c++
- php485函数执行慢怎么优化_php485性能提
- Win10电脑C盘红了怎么清理_Windows10
- 如何在同包不同文件中正确引用 Go 结构体
- MAC如何安装Git版本控制工具_MAC开发环境配

Enumerable
QQ客服