我有一个带有数字的列表,我想使用 LINQ 找到最小值(不是值)的位置
例子:
var lst = new List<int>() { 3, 1, 0, 5 };
现在我正在寻找一个返回我的函数
输出 = 2
因为最小值位于列表中的第 2 位。
var list = new List<int> { 3, 1, 0, 5 };
int pos = list.IndexOf(list.Min()); // returns 2
当您特别要求 LINQ 解决方案时,您得到的只是非 LINQ 解决方案,这里有一个 LINQ 解决方案:
List<int> values = new List<int> { 3, 1, 0, 5 };
int index =
values
.Select((n, i) => new { Value = n, Index = i })
.OrderBy(n=>n.Value)
.First()
.Index;
然而,这并不意味着 LINQ 是解决这个问题的最佳解决方案......
使用更复杂的代码,它的性能会更好一些:
int index =
values
.Select((n, i) => new { Value = n, Index = i })
.Aggregate((a,b) => a.Value < b.Value ? a : b)
.Index;
为了获得最佳性能,您将使用普通循环遍历项目,同时跟踪最低的项目:
int index = 0, value = values[0];
for (int i = 1; i < values.Length; i++) {
if (values[i] < value) {
value = values[i];
index = i;
}
}
捕捉位置的最佳方法是通过FindIndex
此功能仅适用于 List<>
例子
int id = listMyObject.FindIndex(x => x.Id == 15);
如果您有枚举器或数组,请使用这种方式
int id = myEnumerator.ToList().FindIndex(x => x.Id == 15);
或者
int id = myArray.ToList().FindIndex(x => x.Id == 15);
我同意 LINQ 不是解决此问题的最佳解决方案,但这是 O(n) 的另一种变体。它不排序,只遍历列表一次。
var list = new List<int> { 3, 1, 0, 5 };
int pos = Enumerable.Range(0, list.Count)
.Aggregate((a, b) => (list[a] < list[b]) ? a : b); // returns 2
var data = new List<int> { 3, 1, 0, 5 };
var result = Enumerable.Range(0, data.Count).OrderBy(n => data[n]).First();
一个列表可以包含多个等于最小值的元素(见下文)。
我编写的通用扩展方法.FindEveryIndex()
适用于整数、字符串……并且非常灵活,因为您可以将条件指定为 Lambda 表达式。
另一个优点是它返回与条件匹配的所有索引的列表,而不仅仅是第一个元素。
关于您的问题:最小值可以返回为:
var lst = new List<int>() { 1, 2, 1, 3, 4, 1 }; // example list
var minimum = lst.Min(); // get the minumum value of lst
var idx = lst.FindEveryIndex(x => x == minimum); // finds all indices matching condition
Console.WriteLine($"Output: {String.Join(',', idx.ToArray())}"); // show list of indices
它将返回索引 0、2 和 5,因为 in 的最小值lst1
是1
:
输出:0,2,5
示例 2:
void Main()
{
// working with list of integers
var lst1 = new List<int>() { 1, 2, 1, 3, 4, 1 };
lst1.FindEveryIndex(x => x==1).Dump("Find 1"); // finds indices: [0, 2, 5]
lst1.FindEveryIndex(x => x==2).Dump("Find 2"); // finds index: [1]
lst1.FindEveryIndex(x => x==9).Dump("Find 9"); // returns [-1]
// working with list of strings
var lst2 = new List<string>() { "A", "B", "A", "C", "D", "A"};
lst2.FindEveryIndex(x => x=="A").Dump("Find A"); // finds indices: [0, 2, 5]
lst2.FindEveryIndex(x => x=="B").Dump("Find B"); // finds index: [1]
lst2.FindEveryIndex(x => x=="X").Dump("Find X"); // returns [-1]
}
扩展类:
public static class Extension
{
// using System.Collections.Generic;
public static IEnumerable<int> FindEveryIndex<T>(this IEnumerable<T> items,
Predicate<T> predicate)
{
int index = 0; bool found = false;
foreach (var item in items)
{
if (predicate(item))
{
found = true; yield return index;
};
index++;
}
if (!found) yield return -1;
}
}
注意:将这两个代码片段复制到 LinqPad C# 程序中,它可以立即运行。
或者,使用DotNetFiddle在线运行它。
List<int> data = new List<int>();
data.AddRange(new[] { 3, 1, 0, 5 });
Console.WriteLine(data.IndexOf(data.Min()));
int min = 0;
bool minIsSet = false;
var result = ints
.Select( (x, i) => new {x, i}
.OrderBy(z => z.x)
.Select(z =>
{
if (!minIsSet)
{
min = z.x;
minIsSet = true;
}
return z;
}
.TakeWhile(z => z.x == min)
.Select(z => z.i);
我不一定推荐这种 CPS 风格的代码,但它可以工作并且是 O(n),这与使用 OrderBy 的解决方案不同:
var minIndex = list.Aggregate(
new { i = 0, mini = -1, minv = int.MaxValue },
(min, x) => (min.minv > x)
? new { i = min.i + 1, mini = min.i, minv = x }
: new { i = min.i + 1, mini = min.mini, minv = min.minv })
.mini;
如果您想要最后一个最小副本,而不是第一个,请将 > 更改为 >=。
使用 .minv 获取最小值或两者都不获取索引和最小值的 2 元组。
我等不及 .NET 在 4.0 中获得元组。
List<int>.Enumerator e = l.GetEnumerator();
int p = 0, min = int.MaxValue, pos = -1;
while (e.MoveNext())
{
if (e.Current < min)
{
min = e.Current;
pos = p;
}
++p;
}