如何让OrderBy或IComparer对我的特定列表进行排序?

时间:2018-04-07 21:31:20

标签: c# .net sorting dictionary icomparer

字典int我最初使用以下方法在1到100之间的范围内随机生成:

var lstNumbers = Enumerable.Range(1, 100).OrderBy(n => Guid.NewGuid)
                           .GetHashCode()).ToList();

创建字典并用于将ColorType值分配给红色,黄色,白色,红色,黄色,白色等序列中的每个数字到列表中的所有数字:

var startColor = ColorType.Red;
var map = new Dictionary<int, ColorType>();
foreach(var number in lstNumbers) {
  // First color for assignment
  map[number] = startColor.Next();
  // Go to the next color
  startColor = startColor.Next();
}

然后我以几种不同的方式删除项目:

foreach(KeyValuePair<int, ColorType> entry in map.ToList()) {
  if (entry.Key % 2 == 0 && entry.Value == ColorType.Red) {
    map.Remove(entry.Key);
  }
  if (entry.Key % 2 == 1 && entry.Value == ColorType.Yellow) {
    map.Remove(entry.Key);
  }
  if (entry.Key % 3 == 0 && entry.Value == ColorType.White) {
    map.Remove(entry.Key);
  }
}

然后我按升序对数字进行排序:

foreach(var number in map.OrderBy(i => i.Key)) {
  Console.WriteLine(number);
}

Console.ReadLine();

所以我的名单现在看起来很像:

[53, Red]
[54, Yellow]
[55, White]
[56, Yellow]
[61, White]
[62, White]
[64, Yellow]
[65, Red]
ect., ect.

现在我需要按值对这个字典中的最终列表进行排序,其中白色结果位于顶部,黄色位于中间,红色位于底部。红色&lt;黄色&lt;白色,然后显示。

我试过这个OrderBy,但我可能没有正确实现它,或者它可能对我的方案不起作用,因为我收到错误:

  

==运算符不能应用于'ColorType'和'string'的操作数。

foreach(var color in map.OrderBy(c => c.Value == "White" ? 0 : c.Value == "Yellow" ? 1 : 2)) {
  Console.WriteLine(color);
}

我尝试使用IComparer,但我遇到了一个问题,因为我在字典中使用了一种用于颜色的自定义类型:

public class CustomerComparer : IComparer<KeyValuePair<int, ColorType>> {
  private List<string> orderedColors = new List<string>() { "White", "Yellow", "Red" };

  public int Compare(KeyValuePair<int, ColorType> str1, KeyValuePair<int, ColorType> str2) {
    return orderedColors.IndexOf(str1.Value) - orderedColors.IndexOf(str2.Value);
  }
}

var sorted = map.OrderBy(x => x, new CustomerComparer());

foreach(var entry in sorted) {
  Console.WriteLine("{0}: {1}", entry.Key, entry.Value);
}
Console.ReadKey();

我猜测有更好的方法可以做到这一点,另一种方式,或者我完全错过了一些东西。

2 个答案:

答案 0 :(得分:2)

我认为你不需要这里的自定义比较器。您可以使用.ThenBy对订单进行链接。

你可以写

 var sorted = map.OrderBy(c => c.Value != ColorType.White)
                .ThenBy(c => c.Value != ColorType.Yellow)
                .ThenBy(c => c.Value != ColorType.Red);

请注意!=运算符 - 您按布尔值排序,因此false位于true之前。

答案 1 :(得分:2)

如您收到的错误所示,您尝试将ColorTypeString进行比较。你需要使两面相同,要么ColorType

foreach (var color in map.OrderBy(c => c.Value == ColorType.White ?
                                       0 : c.Value == ColorType.Yellow ? 1 : 2)) {
    Console.WriteLine(color);
}

或两者String。您说ColorType是自定义数据类型,因此我不知道ToString()是否会返回颜色名称:

foreach (var color in map.OrderBy(c => c.Value.ToString() == "White" ?
                                       0 : c.Value.ToString() == "Yellow" ? 1 : 2)) {
    Console.WriteLine(color);
}

例如,如果您使用System.Drawing.Color类,ToString()将无效,因为它会返回Color [Red]之类的内容。相反,您可以使用Name属性:

foreach (var color in map.OrderBy(c => c.Value.Name == "White" ?
                                       0 : c.Value.Name == "Yellow" ? 1 : 2)) {
    Console.WriteLine(color);
}

所以我不知道你的ColorType类是否有类似的属性可以返回颜色名称,或者ToString()是否可以完成这项工作。