C#函数式编程是用不可变性、纯函数等思想提升代码质量,LanguageExt库提供Option、Either等类型实现空值和错误的显式处理,推荐渐进式采用。
在 C# 中进行函数式编程,不是要完全抛弃面向对象,而是用不可变性、纯函数、高阶函数、代数数据类型等思想来提升代码的可读性、可测试性和健壮性。LanguageExt 是目前最成熟、被广泛采用的 C# 函数式编程库,它把 Haskell、F# 等语言中经过验证的模式带进了 .NET 生态。
函数式编程的第一步是拥抱不可变性。LanguageExt 提供了 ImmutableArray
List.Add() ,改用 list.Add(item)(返回新列表)Option 替代 null,明确表达“可能没有值”Either 表达“成功或失败”,替代 try/catch 或 out 参数Option
var user = GetUserById(123); // 返回 Optionvar name = user.Map(u => u.Name).IfNone("Anonymous"); // 安全取名 var email = user.Bind(u => u.Email).IfNone("no-email@example.com");
关键点:
Map 对 Some 执行转换,对 None 保持不动Bind(即 flatMap)用于链式调用多个可能失败的操作IfNone 提供默认值,比三元运
算符 + null 检查更语义清晰Either
public static EitherParseInt(string s) => int.TryParse(s, out var i) ? i.AsRight() : "Invalid number".AsLeft(); var result = ParseInt("42").Match( Left: err => Console.WriteLine($"Error: {err}"), Right: val => Console.WriteLine($"Got: {val}") );
好处:
ParseInt(s).Bind(x => DivideBy(x, 0))
Map、Bind、Swap、BiMap 等丰富操作LanguageExt 功能丰富,但不必一步到位。推荐渐进式上手:
Install-Package LanguageExt.Core(基础)+ LanguageExt.CodeGen(如需自动生成不可变 record)Option 和 Either 开始,替换现有 null 和异常逻辑Seq 替代 IEnumerable,获得更函数式的集合 API(如 Filter、Map、Reduce)record class + [Record] 属性生成不可变类型基本上就这些 —— 不复杂,但容易忽略的是:函数式不是炫技,而是让“什么可能出错”“什么可能为空”在类型里一目了然。