LINQ: Как использовать RemoveAll без использования цикла For с массивом - PullRequest
4 голосов
/ 21 мая 2010

В настоящее время у меня есть объект журнала, из которого я хочу удалить объекты на основе запроса LINQ.Я хотел бы удалить все записи в журнале, если сумма версий в программе превышает 60. В настоящее время я вполне уверен, что это сработает, но кажется, что kludgy:

        for (int index = 0; index < 4; index++)
        {
          Log.RemoveAll(log =>
                    (log.Program[index].Version[0].Value +
                     log.Program[index].Version[1].Value +
                     log.Program[index].Version[2].Value ) > 60);
        }

Программа представляет собой массив из 4 значений, а версия имеет массив из 3 значений.Есть ли более простой способ сделать это RemoveAll в LINQ без использования цикла for?

Заранее благодарен за любую помощь!


РЕДАКТИРОВАТЬ: К сожалению, тип переменной, на которой основаны Программа и Версия (это ограничениефреймворк, в котором я работаю) ограничивает нас так, что я не могу получить доступ к любому участнику.Однако я подтвердил, что решение tzaman работает, если у вас есть списки , создав некоторый пример кода.Я ограничен массивоподобными переменными (см. Закомментированные области)

// I'm restricted to Arrays, but if I had lists, this would work. 
  internal class MyLogCollection
  {
    List<MyLog> MyListOfZones = new List<MyLog>();

    public void TestRemove()
    {
      // Original Implementation
      for (int i = 0; i < 4; i++)
      {
        MyListOfZones.RemoveAll(log => (log.MyZoneArray[0].MyVersionArray[0].Value +
                                        log.MyZoneArray[0].MyVersionArray[1].Value +
                                        log.MyZoneArray[0].MyVersionArray[2].Value) > 60);
        //"Any" method is not available off of intellisense scope on MyZoneArray 
      }

      // Better Implementation (thanks tzaman!)
      MyListOfZones.RemoveAll(log => (log.MyZoneArray.Any(prog =>
                                                          prog.MyVersionArray.Sum(ver => ver.Value) > 60)));
    }
  }
  internal class MyLog
  {
    //public MyZone[] MyZoneArray = new MyZone[4];

    public List<MyZone> MyZoneArray = new List<MyZone>(4);

  }
  internal class MyZone
  {
    //public MyVersion[] MyVersionArray = new MyVersion[3];
    public List<MyVersion> MyVersionArray = new List<MyVersion>(3);
  }
  internal class MyVersion
  {
    public byte Value { get; set;}

  }

Спасибо, tzaman!

1 Ответ

8 голосов
/ 21 мая 2010

Это должно сделать это, я думаю:

Log.RemoveAll(log => 
              log.Program.Any(prog => 
                              prog.Version.Sum(ver => ver.Value) > 60));


EDIT: Хорошо, вот как добавить методы расширения, чтобы получить IEnumerable s из ваших индексируемых "массивоподобных" объектов, чтобы вы могли использовать LINQ для них:

static class MyExtensions
{
    public static IEnumerable<MyZone> Enumerate(this MyZoneArray zone)
    {
        for (int i = 0; i < zone.Length; i++)
            yield return zone[i];
    }

    public static IEnumerable<MyVersion> Enumerate(this MyVersionArray version)
    {
        for (int i = 0; i < version.Length; i++)
            yield return version[i]
    }
}

Я предполагаю, что типы MyZoneArray и MyVersionArray имеют поле Length, но если нет, вы можете просто вставить туда 4 и 3. Теперь, когда они установлены, вы можете вызвать функцию Enumerate() для объекта коллекции, чтобы получить версию коллекции IEnumerable со всеми связанными с ней достоинствами LINQy: log.MyZoneArray.Enumerate().Any( ... )

...