如何从.net?</t>中的IEnumerable <t>获取第一个元素

时间:2009-01-30 21:12:22

标签: c# .net

我经常想在.net中抓取IEnumerable<T>的第一个元素,但我还没有找到一个好方法。我提出的最好的是:

foreach(Elem e in enumerable) {
  // do something with e
  break;
}

呸!那么,有一个很好的方法吗?

8 个答案:

答案 0 :(得分:215)

如果你可以使用LINQ,你可以使用:

var e = enumerable.First();

如果enumerable为空,这将抛出异常:在这种情况下,您可以使用:

var e = enumerable.FirstOrDefault();
如果枚举为空,

FirstOrDefault()将返回default(T),对于引用类型,它将是null,对于值类型,将是默认的“零值”。

如果你不能使用LINQ,那么你的方法在技术上是正确的,与使用GetEnumeratorMoveNext方法创建一个枚举器来检索第一个结果没什么不同(这个例子假设可枚举是一个IEnumerable<Elem>):

Elem e = myDefault;
using (IEnumerator<Elem> enumer = enumerable.GetEnumerator()) {
    if (enumer.MoveNext()) e = enumer.Current;
}

Joel Coehoorn在评论中提到.Single();如果你期望你的枚举只包含一个元素,那么它也会起作用 - 但如果它是空的或大于一个元素,它将抛出异常。相应的SingleOrDefault()方法以与FirstOrDefault()类似的方式涵盖此方案。但是,David B解释了SingleOrDefault()在枚举包含多个项目的情况下仍可能抛出异常。

编辑:感谢Marc Gravell指出我需要在使用后处理我的IEnumerator对象 - 我编辑了非LINQ示例以显示using关键字实现这种模式。

答案 1 :(得分:32)

以防您使用的是.NET 2.0并且无法访问LINQ:

 static T First<T>(IEnumerable<T> items)
 {
     using(IEnumerator<T> iter = items.GetEnumerator())
     {
         iter.MoveNext();
         return iter.Current;
     }
 }

这应该做你想要的......它使用泛型,所以你得到IEnumerable的任何类型的第一个项目。

这样称呼:

List<string> items = new List<string>() { "A", "B", "C", "D", "E" };
string firstItem = First<string>(items);

或者

int[] items = new int[] { 1, 2, 3, 4, 5 };
int firstItem = First<int>(items);

你可以很容易地修改它以模仿.NET 3.5的IEnumerable.ElementAt()扩展方法:

static T ElementAt<T>(IEnumerable<T> items, int index)
{
    using(IEnumerator<T> iter = items.GetEnumerator())
    {
        for (int i = 0; i <= index; i++, iter.MoveNext()) ;
        return iter.Current;
    }
} 

这样称呼它:

int[] items = { 1, 2, 3, 4, 5 };
int elemIdx = 3;
int item = ElementAt<int>(items, elemIdx);

当然如果你可以访问LINQ,那么已经发布了很多好的答案......

答案 2 :(得分:24)

好吧,您没有指定您正在使用的.Net版本。

假设你有3.5,另一种方法是ElementAt方法:

var e = enumerable.ElementAt(0);

答案 3 :(得分:6)

FirstOrDefault

Elem e = enumerable.FirstOrDefault();
//do something with e

答案 4 :(得分:0)

如前所述,使用FirstOrDefault或foreach循环。应该避免手动获取枚举器并调用Current。如果它实现IDisposable,foreach将为您处置您的枚举器。当调用MoveNext和Current时,你必须手动处理它(如果适用)。

答案 5 :(得分:0)

如果你的IEnumerable没有暴露它<T>并且Linq失败,你可以使用反射编写一个方法:

public static T GetEnumeratedItem<T>(Object items, int index) where T : class
{
  T item = null;
  if (items != null)
  {
    System.Reflection.MethodInfo mi = items.GetType()
      .GetMethod("GetEnumerator");
    if (mi != null)
    {
      object o = mi.Invoke(items, null);
      if (o != null)
      {
        System.Reflection.MethodInfo mn = o.GetType()
          .GetMethod("MoveNext");
        if (mn != null)
        {
          object next = mn.Invoke(o, null);
          while (next != null && next.ToString() == "True")
          {
            if (index < 1)
            {
              System.Reflection.PropertyInfo pi = o
                .GetType().GetProperty("Current");
              if (pi != null) item = pi
                .GetValue(o, null) as T;
              break;
            }
            index--;
          }
        }
      }
    }
  }
  return item;
}

答案 6 :(得分:0)

试试这个

IEnumberable<string> aa;
string a = (from t in aa where t.Equals("") select t.Value).ToArray()[0];

答案 7 :(得分:0)

您还可以尝试更通用的版本,它会为您提供第i个元素

enumerable.ElementAtOrDefault(I));

希望有所帮助