这是因为字符串的默认排序是标准字母数字字典(词典)排序,ABC11将在ABC2之前,因为排序总是从左到右进行。
要获得您想要的内容,您需要在order by子句中填充数字部分,例如:
var result = partNumbers.OrderBy(x => PadNumbers(x));
其中PadNumbers
可以定义为:
public static string PadNumbers(string input) { return Regex.Replace(input, "[0-9]+", match => match.Value.PadLeft(10, '0')); }
这会为输入字符串中出现的任何数字(或数字)填充零,以便OrderBy
看到:
ABC0000000010 ABC0000000001 ... AB0000000011
填充仅发生在用于比较的密钥上。 原始字符串(没有填充)将保留在结果中。
请注意,此方法假定输入中的数字的最大位数。
在Dave Koelle的网站上可以找到“正常工作”的字母数字排序方法的正确实现。 C#版本就在这里 。
public class AlphanumComparatorFast : IComparer { List GetList(string s1) { List SB1 = new List(); string st1, st2, st3; st1 = ""; bool flag = char.IsDigit(s1[0]); foreach (char c in s1) { if (flag != char.IsDigit(c) || c==''') { if(st1!="") SB1.Add(st1); st1 = ""; flag = char.IsDigit(c); } if (char.IsDigit(c)) { st1 += c; } if (char.IsLetter(c)) { st1 += c; } } SB1.Add(st1); return SB1; } public int Compare(object x, object y) { string s1 = x as string; if (s1 == null) { return 0; } string s2 = y as string; if (s2 == null) { return 0; } if (s1 == s2) { return 0; } int len1 = s1.Length; int len2 = s2.Length; int marker1 = 0; int marker2 = 0; // Walk through two the strings with two markers. List str1 = GetList(s1); List str2 = GetList(s2); while (str1.Count != str2.Count) { if (str1.Count str2[i].ToString().Length) result = -1; else result = 0; } else { int st1ZeroCount=str1[i].ToString().Trim().Length- str1[i].ToString().TrimStart(new char[]{'0'}).Length; int st2ZeroCount = str2[i].ToString().Trim().Length - str2[i].ToString().TrimStart(new char[] { '0' }).Length; if (st1ZeroCount > st2ZeroCount) result = -1; else if (st1ZeroCount
使用此类:
List marks = new List(); marks.Add("M'00Z1"); marks.Add("M'0A27"); marks.Add("M'00Z0"); marks.Add("0000A27"); marks.Add("100Z0"); string[] Markings = marks.ToArray(); Array.Sort(Markings, new AlphanumComparatorFast());
如果你想使用LINQ和一个自定义比较器(如Dave Koelle的自定义比较器)按特定属性对对象列表进行排序,你可以这样做:
... items = items.OrderBy(x => x.property, new AlphanumComparator()).ToList(); ...
您还必须更改Dave的类以inheritanceSystem.Collections.Generic.IComparer
而不是基本IComparer
以便类签名变为:
... public class AlphanumComparator : System.Collections.Generic.IComparer
就个人而言,我更喜欢James McCormack的实现,因为它实现了IDisposable,尽管我的基准测试表明它稍慢。
您可以使用PInvoke获得快速和良好的结果:
class AlphanumericComparer : IComparer { [DllImport("shlwapi.dll", CharSet = CharSet.Unicode)] static extern int StrCmpLogicalW(string s1, string s2); public int Compare(string x, string y) => StrCmpLogicalW(x, y); }
您可以像上面的答案一样使用AlphanumComparatorFast
。
你可以PInvoke到StrCmpLogicalW
(windows函数)来做到这一点。 请参见此处: C#中的自然排序顺序
看起来好像它做了一个字典排序,无论是小型还是大写字母。
您可以尝试在该lambda中使用一些自定义表达式来执行此操作。
在.NET中没有自然的方法可以做到这一点, 但看看这篇关于自然排序的博客文章
您可以将其放入扩展方法并使用它而不是OrderBy
由于开头的字符数是可变的,因此正则表达式有助于:
var re = new Regex(@"d+$"); // finds the consecutive digits at the end of the string var result = partNumbers.OrderBy(x => int.Parse(re.Match(x).Value));
如果有一个固定数量的前缀字符,那么您可以使用Substring
方法从相关字符开始提取:
// parses the string as a number starting from the 5th character var result = partNumbers.OrderBy(x => int.Parse(x.Substring(4)));
如果数字可能包含小数分隔符或千位分隔符,则正则表达式也需要允许这些字符:
var re = new Regex(@"[d,]*.?d+$"); var result = partNumbers.OrderBy(x => double.Parse(x.Substring(4)));
如果正则表达式或Substring
串返回的Substring
可能由int.Parse
/ double.Parse
无法解析,则使用相关的TryParse
变量:
var re = new Regex(@"d+$"); // finds the consecutive digits at the end of the string var result = partNumbers.OrderBy(x => { int? parsed = null; if (int.TryParse(re.Match(x).Value, out var temp)) { parsed = temp; } return parsed; });
我不知道如何在LINQ中做到这一点,但也许你喜欢这样:
Array.Sort(partNumbers, new AlphanumComparatorFast());
//显示结果
上述就是C#学习教程:使用LINQ进行字母数字排序分享的全部内容,如果对大家有所用处且需要了解更多关于C#学习教程,希望大家多多关注—编程笔记
foreach (string h in partNumbers ) { Console.WriteLine(h); }