Условный выбор в LINQ (выберите вместо, если он пуст) - PullRequest
6 голосов
/ 03 апреля 2012

Есть ли какой-нибудь способ LINQ для условного выбора данных, то есть выбора из другого источника, если первый был пуст?Например, если у вас есть древовидная структура элементов, и вы хотите получить какой-либо актив из корня или из пустых его дочерних элементов.

У меня есть следующий пример:

IEnumerable<Item> items = ...;
// Item has a Assets property that returns IEnumerable<Asset>
// Item has a SubItems property that returns IEnumerable<Item>
//    i.e. other items with assets in them

// getting assets from a "main" item
var assets = item.Assets.Where(a => HasRelevantAsset(a));

// if there were no relevant assets in the "main" item
if (!assets.Any()) {
    // then reselect from "subitems" assets instead
    assets = item.SubItems.SelectMany(item => 
        item.Assets.Where(a => HasRelevantAsset(a)));
}

// HasRelevantAsset(Asset) is a static method that returns 
// true if it is the asset that is needed

1 Ответ

1 голос
/ 03 апреля 2012

Я верю, что LINQ будет выглядеть немного некрасиво

var assets = item.Any(a=>HaRelevantAsset(a)) ? item.Where(a => HasRelevantAsset(a)) :
                     item.SubItems.SelectMany(item => 
                            item.Assets.Where(a => HasRelevantAsset(a)));

Я бы выбрал другой вариант, метод расширения

public static IEnumerable<Asset> SelectRelevantAssets(this Item item)
{
     var assetsInItemFound = false;
     foreach (var asset in item.Assets)
     {
         if (HasRelevantAsset(asset))
         {
             assetsInItemFound = true;
             yield return asset;
         }
     }
     if (assetsInItemFound)
     {
         yield break;
     }
     else
     {
         foreach (var subItem in item.SubItems)         
             foreach (var asset in subItem.Assets)
                if (HasRelevantAsset(asset))
                    yield return asset;
     }
}





Сначала я хотел попробовать рекурсивный вызов SelectRelevantAssets, я думаю, что это будет похоже на

if (!assetsInItemFound)
         {
             yield break;
         }
         else
         {
             foreach (var subItem in item.SubItems)         
                 foreach (var asset in SelectRelevantAssets(subItem))
                        yield return asset;
         }

Но это будет включать в себя активы, найденные в коллекции предметов активов подэлемента

...