Count() 最快但需防 null,仅支持单字符统计;忽略大小写用 Char.ToLower();子串统计须用 IndexOf 循环;性能敏感时优先 for 循环。
用 Count() 最快,但要注意字符串为空或 null
直接调用 string.count() 是最简洁的方式,前提是字符串不为 null。它底层是遍历一次,时间复杂度 o(n),没有额外分配内存。
- 如果
str是null,会抛出NullReferenceException—— 实际项目里必须先判空或用空合并操作符 - 区分大小写:默认严格匹配,
'a'和'A'算不同字符 - 不能直接传入子串(比如
"ab"),只支持单个char
示例:
string str = "hello world";<br>int count = str?.Count(c => c == 'l') ?? 0; // 返回 3
想忽略大小写?别手写循环,用 Char.ToLower() 统一转换
LINQ 本身不提供内置的忽略大小写选项,硬写 c == 'a' || c == 'A' 可读性差且易漏。更稳妥的做法是在比较前统一转小写(或大写)。
-
Char.ToLower(c)比c.ToString().ToLower()高效得多,避免字符串分配 - 注意某些 Unicode 字符在不同文化下可能有特殊映射,普通场景用默认
CultureInfo.InvariantCulture就够 - 别用
String.Equals(c.ToString(), "a", StringComparison.OrdinalIgnoreCase)—— 开销太大,每次都在创建临时字符串
示例:
int count = str?.Count(c => Char.ToLower(c) == 'a') ?? 0;
统计子字符串(如 “ab”)出现次数?Count() 不行,得换思路
string.Count(c => ……) 只能处理单字符,遇到子串必须改用其他方法。常见错误是用 IndexOf 循环但没处理重叠匹配(比如在 "aaaa" 中找 "aa",重叠时应得 3 次,非重叠才是 2 次)。
- 明确你要的是「重叠」还是「非重叠」匹配 —— 这直接影响实现逻辑
- 用
IndexOf(string, startIndex)循环是最轻量的选择,比正则快,也比Split()更省内存 - 别用
str.Replace(target, "").Length算差值 —— 创建了新字符串,对长文本性能差,且无法处理重叠
示例(非重叠):
string s = "ababab";<br>string sub = "ab";<br>int count = 0, i = 0;<br>while ((i = s.IndexOf(sub, i)) != -1) {count++; i += sub.Length;}
性能敏感场景下,别依赖 LINQ —— 原生 for 循环反而更稳
当字符串超长(比如几 MB 日志文本)、或在 tight loop 里高频调用时,Count() 的委托调用开销和枚举器分配会显现出来。此时裸写 for 循环不仅更快,还更容易控制边界和提前退出。
- foreach / LINQ 在 .NET 6+ 有部分优化,但依然不如直接索引访问
- 如果只需要判断「是否至少出现一次」,用
Contains()或手动 break,别硬 Count 到最后 - 注意
string.Length是 O(1),放心用;别为了“省一次属性访问”缓存它,编译器通常会帮你优化
示例:
int count = 0;<br>for (int i = 0; i < str.Length; i++) {if (str[i] == 'x') count++; }
最容易被忽略的是 null 安全和子串匹配语义 —— 同一个需求,在日志分析、用户输入校验、协议解析里,对“重复”“重叠”“大小写”的要求可能完全不同,别套模板代码。