提高linq查询的性能

时间:2012-12-14 11:52:04

标签: c# performance linq

我正在使用一些Linq查询优化方法。到目前为止,执行时间大约是3秒,我正在尝试减少它。该方法中发生了大量的操作和计算,但没有太复杂。

我将非常感谢如何改进性能和代码优化的任何建议和想法。


该方法的整个代码(下面我将指出我有最大延迟的地方):

public ActionResult DataRead([DataSourceRequest] DataSourceRequest request)
{
    CTX.Configuration.AutoDetectChangesEnabled = false;

    var repoKomfortaktion = new KomfortaktionRepository();
    var komfortaktionen = CTX.Komfortaktionen.ToList();

    var result = new List<AqGeplantViewModel>();

    var gruppen = new HashSet<Guid?>(komfortaktionen.Select(c => c.KomfortaktionsGruppeId).ToList());

    var hochgeladeneKomplettabzuege = CTX.Komplettabzug.Where(c => gruppen.Contains(c.KomfortaktionsGruppeId)).GroupBy(c => new { c.BetriebId, c.KomfortaktionsGruppeId }).Select(x => new { data = x.Key }).ToList();
    var teilnehmendeBetriebe = repoKomfortaktion.GetTeilnehmendeBetriebe(CTX, gruppen);
    var hochgeladeneSperrlistenPlz = CTX.SperrlistePlz.Where(c => gruppen.Contains(c.KomfortaktionsGruppeId) && c.AktionsKuerzel != null)
                                                                        .GroupBy(c => new { c.AktionsKuerzel, c.BetriebId, c.KomfortaktionsGruppeId }).Select(x => new { data = x.Key }).ToList();

    var hochgeladeneSperrlistenKdnr = CTX.SperrlisteKdnr.Where(c => gruppen.Contains(c.KomfortaktionsGruppeId) && c.AktionsKuerzel != null)
                                                                        .GroupBy(c => new { c.AktionsKuerzel, c.BetriebId, c.KomfortaktionsGruppeId }).Select(x => new { data = x.Key }).ToList();

    var konfigsProAktion = CTX.Order.GroupBy(c => new { c.Vfnr, c.AktionsId }).Select(c => new { count = c.Count(), c.Key.AktionsId, data = c.Key }).ToList();
    foreach (var komfortaktion in komfortaktionen)
    {
        var item = new AqGeplantViewModel();

        var zentraleTeilnehmer = teilnehmendeBetriebe.Where(c => c.TeilnahmeStatus.Any(x => x.KomfortaktionId == komfortaktion.Id && x.AktionsTypeId == 1)).ToList();
        var lokaleTeilnehmer = teilnehmendeBetriebe.Where(c => c.TeilnahmeStatus.Any(x => x.KomfortaktionId == komfortaktion.Id && x.AktionsTypeId == 2)).ToList();

        var hochgeladeneSperrlistenGesamt =
                hochgeladeneSperrlistenPlz.Count(c => c.data.AktionsKuerzel == komfortaktion.Kuerzel && c.data.KomfortaktionsGruppeId == komfortaktion.KomfortaktionsGruppeId) +
                hochgeladeneSperrlistenKdnr.Count(c => c.data.AktionsKuerzel == komfortaktion.Kuerzel && c.data.KomfortaktionsGruppeId == komfortaktion.KomfortaktionsGruppeId);

        item.KomfortaktionId = komfortaktion.KomfortaktionId;
        item.KomfortaktionName = komfortaktion.Aktionsname;
        item.Start = komfortaktion.KomfortaktionsGruppe.StartAdressQualifizierung.HasValue ? komfortaktion.KomfortaktionsGruppe.StartAdressQualifizierung.Value.ToString("dd.MM.yyyy") : string.Empty;
        item.LokalAngemeldet = lokaleTeilnehmer.Count();
        item.ZentralAngemeldet = zentraleTeilnehmer.Count();

        var anzHochgelandenerKomplettabzuege = hochgeladeneKomplettabzuege.Count(c => zentraleTeilnehmer.Count(x => x.BetriebId == c.data.BetriebId) == 1) +
                                               hochgeladeneKomplettabzuege.Count(c => lokaleTeilnehmer.Count(x => x.BetriebId == c.data.BetriebId) == 1);

        item.KomplettabzugOffen = (zentraleTeilnehmer.Count() + lokaleTeilnehmer.Count()) - anzHochgelandenerKomplettabzuege;

        item.SperrlisteOffen = (zentraleTeilnehmer.Count() + lokaleTeilnehmer.Count()) - hochgeladeneSperrlistenGesamt;

        item.KonfigurationOffen = zentraleTeilnehmer.Count() - konfigsProAktion.Count(c => c.AktionsId == komfortaktion.KomfortaktionId && zentraleTeilnehmer.Any(x => x.Betrieb.Vfnr == c.data.Vfnr));

        item.KomfortaktionsGruppeId = komfortaktion.KomfortaktionsGruppeId;
        result.Add(item);
    }

    return Json(result.ToDataSourceResult(request));
}

上半场(在foreach之前)需要半秒钟,这是可以的。最大的延迟是在第一次迭代中的foreach语句中,特别是在这些行中,zentraleTeilnehmer的执行首次需要1.5秒。

var zentraleTeilnehmer = teilnehmendeBetriebe.Where(c => c.TeilnahmeStatus.Any(x => x.KomfortaktionId == komfortaktion.Id && x.AktionsTypeId == 1)).ToList();
var lokaleTeilnehmer = teilnehmendeBetriebe.Where(c => c.TeilnahmeStatus.Any(x => x.KomfortaktionId == komfortaktion.Id && x.AktionsTypeId == 2)).ToList();

TeilnehmendeBetriebe拥有超过800条线路,TeilnahmeStatus酒店通常有4条线路。所以,最多800 * 4次迭代,这毕竟不是一个庞大的数字...

因此,我主要是在优化这些线路上,希望将执行时间缩短到半秒左右。

我尝试了什么:

  1. 将Linq重写为foreach:没有帮助,同时......可能并不奇怪,但值得一试。

    foreach (var tb in teilnehmendeBetriebe) //836 items
    {
        foreach (var ts in tb.TeilnahmeStatus) //3377 items
        {
            if (ts.KomfortaktionId == komfortaktion.Id && ts.AktionsTypeId == 1)
            {
                testResult.Add(tb);
                break;
            }
        }
    }
    
  2. 使用.Select()选择teilnehmendeBetriebe的特定列。也没有帮助。

  3. 我没有尝试过其他小动作。

    有趣的是 - 虽然foreach的第一次迭代可能需要2秒钟,而第二次迭代只需要使用milisecons,因此.net能够优化或重用计算数据。

    非常欢迎任何关于可以改变什么以提高性能的建议!

    编辑:
    TeilnahmeBetriebKomfortaktion.TeilnahmeStatus方法GetTeilnehmendeBetriebe急切地加载 public List<TeilnahmeBetriebKomfortaktion> GetTeilnehmendeBetriebe(Connection ctx, HashSet<Guid?> gruppen) { return ctx.TeilnahmeBetriebKomfortaktion.Include( c => c.TeilnahmeStatus).ToList(); }

        SELECT 
        [Extent1].[Id] AS [Id], 
        [Extent1].[BetriebId] AS [BetriebId], 
        [Extent1].[MandantenId] AS [MandantenId], 
        [Extent1].[CreatedUser] AS [CreatedUser], 
        [Extent1].[UpdatedUser] AS [UpdatedUser], 
        [Extent1].[CreatedDate] AS [CreatedDate], 
        [Extent1].[UpdatedDate] AS [UpdatedDate], 
        [Extent1].[IsDeleted] AS [IsDeleted]
        FROM [Semas].[TeilnahmeBetriebKomfortaktion] AS [Extent1]
        WHERE [Extent1].[IsDeleted] <> cast(1 as bit)
    

    EDIT2: 执行GetTeilnehmendeBetriebe时发送的查询:

    {{1}}

1 个答案:

答案 0 :(得分:2)

我的假设是TeilnahmeBetriebKomfortaktion.TeilnahmeStatus是一个延迟加载的集合,导致N + 1 problem。您应该急切地获取该集合以提高性能。

foreach循环的以下迭代很快,因为在第一次迭代之后,不再从数据库服务器请求这些对象,而是来自内存的服务器。