2012-03-04 96 views
2

我需要加載一堆用於生成報表的數據。涉及圖中大約8或9種不同的實體類型。實體框架批量顯式加載

如果我打電話Include包括我需要的所有數據,結果查詢是如此複雜(左連接,聯合,case語句嘉豪),它需要大約600秒執行。

如果我讓延遲加載爲我生成報表的延遲加載數據,大約需要180秒來創建報表。更好,但仍然不能接受。

如果我做了

「加載基於相關實體 類型的ID的下一個實體型」

辦法(有點像LLBLGEN如果你熟悉它),我可以在大約3秒鐘內獲得所需的全部數據。

有關如何做到這一點的任何建議?

基本上我想要做這樣的事情:

var invoices = objectContext.Invoices.Where(...reportCriteria...).ToArray(); 

objectContext.LoadProperty(invoices, "InvoiceReceivables"); 

objectContext.LoadProperty(invoices.SelectMany(i => i.InvoiceReceivables), "Adjustments"); 

objectContext.LoadProperty(invoices.SelectMany(i => i.InvoiceReceivables.SelectMany(ir => ir.Adjustments)), "AdjustmentComments") 

... and so on 

但LoadProperty只適用於單一的實體,而不是收藏。

除了自己做查詢和構建對象圖以外,還有什麼辦法嗎?

回答

2

Althugh E.J.的回答非常有效,可能是一種更好,更高效的方法,我們現在沒有真正的帶寬來重構報告。我把這個放在一起,它似乎在伎倆。也許它會對其他人有一些好處......爲簡潔起見,省略了一些簡單的輔助方法。

用法:

Q.GetQueryableFactory(objectContext).Load(invoices, 
    i => i.InvoiceReceivables.SelectMany(ir => ir.Adjustments.SelectMany(
     a => a.AdjustmentComments))) 


public static class Q 
{ 
    /// <summary> 
    /// Gets a queryable factory that returns a queryable for a specific entity type. 
    /// </summary> 
    /// <param name="objectContext">The object context.</param> 
    /// <returns></returns> 
    public static Func<Type, IQueryable> GetQueryableFactory(object objectContext) 
    { 
     var queryablePropertiesByType = objectContext.GetType().GetProperties().Where(p => p.GetIndexParameters().Length == 0 && p.PropertyType.IsGenericTypeFor(typeof(IQueryable<>))) 
      .ToDictionary(p => p.PropertyType.FindElementType()); 

     return t => 
        { 
         PropertyInfo property; 
         if (!queryablePropertiesByType.TryGetValue(t, out property)) 
         { 
          property = queryablePropertiesByType.Values.FirstOrDefault(p => p.PropertyType.FindElementType().IsAssignableFrom(t)) 
           .EnsureNotDefault("Could not find queryable for entity type {0}.".FormatWith(t.Name)); 

          var queryable = property.GetValue(objectContext, null); 

          return (IQueryable)typeof(System.Linq.Queryable).GetMethod("OfType").MakeGenericMethod(t).Invoke(null, new[] { queryable }); 
         } 

         return (IQueryable)property.GetValue(objectContext, null); 
        }; 
    } 

    /// <summary> 
    /// Loads the target along the specified path, using the provided queryable factory. 
    /// </summary> 
    /// <typeparam name="T"></typeparam> 
    /// <param name="queryableFactory">The queryable factory.</param> 
    /// <param name="target">The target.</param> 
    /// <param name="path">The path.</param> 
    public static void Load<T>(this Func<Type, IQueryable> queryableFactory, T target, Expression<Func<T, object>> path) 
    { 
     queryableFactory.Load(target, path.AsEnumerable().Reverse().OfType<MemberExpression>().Select(m => m.Member.Name).Join(".")); 
    } 

    /// <summary> 
    /// Loads the target along the specified path, using the provided queryable factory. 
    /// </summary> 
    /// <typeparam name="T"></typeparam> 
    /// <param name="queryableFactory">The queryable factory.</param> 
    /// <param name="target">The target.</param> 
    /// <param name="path">The path.</param> 
    public static void Load<T>(this Func<Type, IQueryable> queryableFactory, IEnumerable<T> target, Expression<Func<T, object>> path) 
    { 
     queryableFactory.Load(target, path.ToMemberAccessStrings().First()); 
    } 

    /// <summary> 
    /// Loads the target along the specified path, using the provided queryable factory. 
    /// </summary> 
    /// <param name="queryableFactory">The queryable factory.</param> 
    /// <param name="target">The target.</param> 
    /// <param name="path">The path.</param> 
    public static void Load(this Func<Type, IQueryable> queryableFactory, object target, string path) 
    { 
     foreach (var pathPart in path.Split('.')) 
     { 
      var property = (target.GetType().FindElementType() ?? target.GetType()).GetProperty(pathPart); 

      LoadProperty(queryableFactory(property.PropertyType.FindElementType() ?? property.PropertyType), target, pathPart); 

      if (target is IEnumerable) 
      { 
       // select elements along path target.Select(i => i.Part).ToArray() 
       target = target.CastTo<IEnumerable>().AsQueryable().Select(pathPart).ToInferredElementTypeArray(); 

       var propertyElementType = property.PropertyType.FindElementType(); 
       if (propertyElementType != null) 
       { 
        target = target.CastTo<object[]>().SelectMany(i => i.CastTo<IEnumerable>().ToObjectArray()).ToArray(propertyElementType); 
       } 
      } 
      else 
      { 
       target = property.GetValue(target, null); 
      } 
     } 
    } 

    /// <summary> 
    /// Loads the property on the target using the queryable source. 
    /// </summary> 
    /// <param name="source">The source.</param> 
    /// <param name="target">The target.</param> 
    /// <param name="targetProperty">The target property.</param> 
    /// <param name="targetIdProperty">The target id property.</param> 
    /// <param name="sourceProperty">The source property.</param> 
    public static void LoadProperty(this IQueryable source, object target, string targetProperty, string targetIdProperty = null, string sourceProperty = null) 
    { 
     var targetType = target.GetType(); 
     targetType = targetType.FindElementType() ?? (targetType.Assembly.IsDynamic && targetType.BaseType != null ? targetType.BaseType : targetType); 

     // find the property on the source so we can do queryable.Where(i => i.???) 
     var sourceType = source.ElementType; 
     PropertyInfo sourcePropertyInfo; 
     if (sourceProperty == null) 
     { 
      sourcePropertyInfo = sourceType.GetProperty(targetType.Name + "Id") ?? sourceType.GetProperty(targetType.Name + "ID") ?? sourceType.GetProperty("Id") ?? sourceType.GetProperty("ID"); 
     } 
     else 
     { 
      sourcePropertyInfo = sourceType.GetProperty(sourceProperty); 
     } 

     if (sourcePropertyInfo == null || sourcePropertyInfo.GetIndexParameters().Length != 0) throw new InvalidOperationException("Could not resolve id property on source {0}.".FormatWith(source.ElementType.Name)); 


     // find the property on the target so we can find the relevant source objects via queryable.Where(i => i.Property == ???) 
     PropertyInfo targetIdPropertyInfo; 
     if (targetIdProperty == null) 
     { 
      targetIdPropertyInfo = targetType.GetProperty(targetProperty + "Id") ?? targetType.GetProperty(targetProperty + "ID") ?? targetType.GetProperty("Id") ?? targetType.GetProperty("Id").EnsureNotDefault("Could not resolve id property to use on {0}.".FormatWith(targetType.Name)); 
     } 
     else 
     { 
      targetIdPropertyInfo = targetType.GetProperty(targetIdProperty); 
     } 

     if (targetIdPropertyInfo == null || targetIdPropertyInfo.GetIndexParameters().Length != 0) throw new InvalidOperationException("Could not resolve id property for {0} on target {1}.".FormatWith(targetProperty, targetType.Name)); 


     var targetPropertyInfo = targetType.GetProperty(targetProperty); 
     if (targetPropertyInfo == null || targetPropertyInfo.GetIndexParameters().Length != 0) throw new InvalidOperationException("Could not find property {0} on target type {1}.".FormatWith(targetProperty, target.GetType())); 

     // go get the data and set the results. 
     if (target is IEnumerable) 
     { 
      // filter to only non loaded targets 
      var nullOrEmptyPredicate = "{0} == null".FormatWith(targetPropertyInfo.Name); 
      if (targetPropertyInfo.PropertyType.FindElementType() != null) nullOrEmptyPredicate += " or {0}.Count = 0".FormatWith(targetPropertyInfo.Name); 
      target = target.CastTo<IEnumerable>().AsQueryable().Where(nullOrEmptyPredicate).ToInferredElementTypeArray(); 

      var ids = target.CastTo<IEnumerable>().OfType<object>().Select(i => targetIdPropertyInfo.GetValue(i, null)).WhereNotDefault().Distinct().ToArray(); 

      if (!ids.Any()) return; 

      var predicate = ids.Select((id, index) => "{0} = @{1}".FormatWith(sourcePropertyInfo.Name, index)).Join(" or "); 
      // get the results in one shot 
      var results = source.Where(predicate, ids.ToArray()).ToInferredElementTypeArray().AsQueryable(); 

      foreach (var targetItem in target.CastTo<IEnumerable>()) 
      { 
       SetResultsOnTarget(results, targetItem, sourcePropertyInfo, targetIdPropertyInfo, targetPropertyInfo); 
      } 
     } 
     else 
     { 
      // only fetch if not loaded already 
      var value = targetPropertyInfo.GetValue(target, null); 
      if (value == null || value.As<IEnumerable>().IfNotNull(e => e.IsNullOrEmpty())) 
      { 
       SetResultsOnTarget(source, target, sourcePropertyInfo, targetIdPropertyInfo, targetPropertyInfo); 
      } 
     } 

    } 

    /// <summary> 
    /// Sets the results on an individual target entity. 
    /// </summary> 
    /// <param name="source">The source.</param> 
    /// <param name="target">The target.</param> 
    /// <param name="sourcePropertyInfo">The source property info.</param> 
    /// <param name="targetIdPropertyInfo">The target id property info.</param> 
    /// <param name="targetPropertyInfo">The target property info.</param> 
    private static void SetResultsOnTarget(IQueryable source, object target, PropertyInfo sourcePropertyInfo, PropertyInfo targetIdPropertyInfo, PropertyInfo targetPropertyInfo) 
    { 
     var id = targetIdPropertyInfo.GetValue(target, null); 

     var results = source.Where("{0} = @0".FormatWith(sourcePropertyInfo.Name), id).As<IEnumerable>().OfType<object>().ToArray(); 

     var targetPropertyElementType = targetPropertyInfo.PropertyType.FindElementType(); 
     if (targetPropertyElementType != null) 
     { 
      // add all results 
      object collection = targetPropertyInfo.GetValue(target, null); 

      if (collection == null) 
      { 
       // instantiate new collection, otherwise use existing 
       collection = Activator.CreateInstance(typeof(List<>).MakeGenericType(targetPropertyElementType)); 
       targetPropertyInfo.SetValue(target, collection, null); 
      } 

      var addMethod = collection.GetType().GetMethods().FirstOrDefault(m => m.Name == "Add" && m.GetParameters().Length == 1 && m.GetParameters()[0].ParameterType.IsAssignableFrom(targetPropertyElementType)).EnsureNotDefault("Could not find add method for collection type."); 

      foreach (var result in results) 
      { 
       if (!Enumerable.Contains((dynamic)collection, result)) addMethod.Invoke(collection, new[] { result }); 
      } 
     } 
     else 
     { 
      targetPropertyInfo.SetValue(target, results.FirstOrDefault(), null); 
     } 
    } 
} 
1

我使用直接EF4.0爲我的應用程序的UI方面的所有數據訪問,但是當涉及到報表時,我幾乎總是放棄'EF方式',並針對存儲過程和/或視圖,因此所有複雜和耗時的邏輯/彙總都可以及時在DB服務器中完成。這樣的表現總是很棒。

有些事情要考慮,如果你不能以足夠快的速度獲得EF來滿足你的需求。