1

我正在尝试将下面嵌套的每个循环转换为 Linq。但是我仍然无法成功。

var objAct = new List<InformaticsBenchmarkSummary>();
foreach (var item in op)
{
    foreach (var lstTp5 in lstTopFive)
    {
        if (item.UnitOfOperations.ContainsKey(lstTp5.SystemID))
        {
            var objIbm = new InformaticsBenchmarkSummary();
            objIbm.CompanyId = item.CompanyId;
            objIbm.CompanyName = item.CompanyName;
            objIbm.LocationId = item.LocationId;
            objIbm.LocationName = item.LocationName;
            objIbm.UnitOfOperations.Add(lstTp5.SystemID, 
                                        item.UnitOfOperations[lstTp5.SystemID]);
            objAct.Add(objIbm);
        }
    }
}

UnitOfOperations类型在哪里Dictionary<int,string>()
opList<InformaticsBenchmarkSummary>()
lstTopFiveList<int>()

我试过了,类似这样的东西,但在语法上不成功

var output = from item in op
from lstTp5 in lstTopFive
where item.UnitOfOperations.ContainsKey(lstTp5.SystemID)
let v = new InformaticsBenchmarkSummary()
{
     CompanyId = item.CompanyId,
     CompanyName = item.CompanyName,
     LocationId = item.LocationId,
     LocationName = item.LocationName
}
.UnitOfOperations.Add(lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID])
select v;

嵌套循环完美地工作,但我认为,对此的 linq 会提高性能。感谢任何帮助。

4

4 回答 4

2

在 Linq 查询语法中不可能在UnitOfOperations.Add选择中使用 。但是您可以使用方法链和SelectMany方法来做到这一点:

var objAcu = (op.SelectMany(item => lstTopFive, (item, lstTp5) => new { item, lstTp5 })  // <- Bad readability
                .Where(t => t.item.UnitOfOperations.ContainsKey(t.lstTp5.SystemID))
                .Select(t =>
                        {
                            var objIbm = new InformaticsBenchmarkSummary
                            {
                                CompanyId = t.item.CompanyId,
                                CompanyName = t.item.CompanyName,
                                LocationId = t.item.LocationId,
                                LocationName = t.item.LocationName
                            };
                            objIbm.UnitOfOperations.Add(t.lstTp5.SystemID, t.item.UnitOfOperations[t.lstTp5.SystemID]);
                            return objIbm;
                        })).ToList();

如果属性UnitOfOperations具有 public set,在这种情况下,您可以使用查询语法。

你如何替换 1 foreach?由一个from ... in ...

然后,要替换 2 foreach,请使用 2from ... in ...

var objAct = (from item in op  // First foreach loop
              from lstTp5 in lstTopFive  // Second foreach loop
              where item.UnitOfOperations.ContainsKey(lstTp5.SystemID)
              select new InformaticsBenchmarkSummary
              {
                  CompanyId = item.CompanyId,
                  CompanyName = item.CompanyName,
                  LocationId = item.LocationId,
                  LocationName = item.LocationName,
                  UnitOfOperations = { { lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID] } }
              }).ToList();

但我怀疑它会提高这种操作的性能。

无论如何,我不明白你试图达到什么目的。因为在输出的所有项目中,字典中都会有一个且只有一个元素UnitOfOperations。真的是你想做的吗?

更新

List<int> systemIdTop5 = lstTopFive.Select(tp5 => tp5.SystemID).ToList();
var objAct = (from item in op
              select new InformaticsBenchmarkSummary
              {
                  CompanyId = item.CompanyId,
                  CompanyName = item.CompanyName,
                  LocationId = item.LocationId,
                  LocationName = item.LocationName,
                  UnitOfOperations = systemIdTop5.Intersect(item.UnitOfOperations.Keys)
                                                 .ToDictionary(systemId => systemId, systemId => item.UnitOfOperations[systemId])
              }).ToList();
于 2013-02-05T11:38:09.107 回答
1

你很接近,但你不能void像这样在 LINQ 查询中使用返回方法。(如果它不是void-return,那么v将是 的结果Add(),这很可能是错误的。)

如果你想创建一个新Dictionary的 for UnitOfOperations,你可以像其他属性一样设置它。但是如果你不能这样做(可能是因为UnitOfOperations有一个私有设置器)或者你不想这样做(因为UnitOfOperations被初始化为你想要保留的某个值),你可以使用 C# 的一个鲜为人知的特性:里面的集合初始化器对象初始化器:

UnitOfOperations = { { lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID] } }

这段代码的效果和你写的一样:

createdObject.UnitOfOperations.Add(lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID]);

唯一的区别是它不是语句,它是表达式的一部分,这意味着您可以在 LINQ 查询中使用它。

整个查询将是:

var output = from item in op
             from lstTp5 in lstTopFive
             where item.UnitOfOperations.ContainsKey(lstTp5.SystemID)
             select new InformaticsBenchmarkSummary()
             {
                 CompanyId = item.CompanyId,
                 CompanyName = item.CompanyName,
                 LocationId = item.LocationId,
                 LocationName = item.LocationName,
                 UnitOfOperations =
                 {
                     { lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID] }
                 }
             };
于 2013-02-05T11:57:12.357 回答
1

据我所知,您遇到困难的是您的 UnitOfOperations。如果它在构造函数中被初始化,你可以使用这个:

            var output = from item in op
                     from lstTp5 in lstTopFive
                     where item.UnitOfOperations.ContainsKey(lstTp5.SystemID)
                     select
                         new InformaticsBenchmarkSummary()
                             {
                                 CompanyId = item.CompanyId,
                                 CompanyName = item.CompanyName,
                                 LocationId = item.LocationId,
                                 LocationName = item.LocationName,
                                 UnitOfOperations = { { lstTp5.SystemID, item.UnitOfOperations[lstTp5.SystemID] } }
                             };

结果是一个IEnumerable,如果你想要它作为一个列表,调用output.ToList()

两个旁注:

  1. 我不相信这会更快。它仍然是一个内部循环。
  2. 这可能会在结果中产生几乎重复的项目(不同UnitOfOperations),但我想这是需要的。在最坏的情况下,in 中的所有项目op都有一个UnitOfOperations包含所有SystemIDin的项目,lstTopFive从而为我们提供了 中的op.Count()*lstTopFive.Count()项目总数output
于 2013-02-05T12:19:18.370 回答
0

可以帮助你:

var output = from item in op
                 join lstTp5 in lstTopFive on item.UnitOfOperations.Key equals lstTp5.SystemID
                 select new InformaticsBenchmarkSummary
                 {
                     CompanyId = item.CompanyId,
                     CompanyName = item.CompanyName,
                     LocationId = item.LocationId,
                     LocationName = item.LocationName,
                     UnitOfOperations = item.UnitOfOperations 
                 };
于 2013-02-12T04:41:03.390 回答