最近我在最佳化 Include 拆分查詢,貼出原始碼供大家交流探討是否還有最佳化空間。
測試程式碼
1 Console.WriteLine($"總記錄數:{db.Query<Category>().Count()}"); 2 3 var stopwatch1 = new Stopwatch(); 4 stopwatch1.Start(); 5 var data1 = db.Query<Category>().Include(i => i.Products).ToList(); 6 stopwatch1.Stop(); 7 8 Console.WriteLine($"Include查詢 耗時:{stopwatch1.ElapsedMilliseconds} ms {stopwatch1.ElapsedMilliseconds / 1000.00}s"); 9 10 //Console.WriteLine(Json.Serialize(data1[0].Products[0])); 11 12 var stopwatch2 = new Stopwatch(); 13 stopwatch2.Start(); 14 var data2 = db.Query<Category>().ToList(); 15 16 foreach (var item in data2) 17 { 18 item.Products = db.Query<Product>().Where(w => w.CategoryId == item.CategoryId).ToList(); 19 } 20 stopwatch2.Stop(); 21 22 Console.WriteLine($"迴圈查詢 耗時:{stopwatch2.ElapsedMilliseconds} ms {stopwatch2.ElapsedMilliseconds / 1000.00}s");
測試結果
Include 生成的Sql語句
SELECT `CategoryId`,`CategoryName` FROM `Category` -------------------------- -------------------------- SELECT a.`CategoryId`,a.`CategoryName`,b.`ProductId`,b.`CategoryId`,b.`ProductCode`,b.`ProductName`,b.`DeleteMark`,b.`CreateTime`,b.`Custom1`,b.`Custom2`,b.`Custom3`,b.`Custom4`,b.`Custom5`,b.`Custom6`,b.`Custom7`,b.`Custom8`,b.`Custom9`,b.`Custom10`,b.`Custom11`,b.`Custom12` FROM `Category` `a` INNER JOIN `Product` `b` ON `a`.`CategoryId` = `b`.`CategoryId` ORDER BY `b`.`ProductId`
Include 方法實現
1 /// <summary> 2 /// 包括 3 /// </summary> 4 /// <typeparam name="TProperty"></typeparam> 5 /// <param name="expression">表示式</param> 6 /// <returns></returns> 7 public IInclude<T, TProperty> Include<TProperty>(Expression<Func<T, TProperty>> expression) where TProperty : class 8 { 9 var result = expression.ResolveSql(new ResolveSqlOptions() 10 { 11 DbType = ado.DbOptions.DbType, 12 ResolveSqlType = ResolveSqlType.NewColumn, 13 IgnoreParameter = true, 14 IgnoreIdentifier = true 15 }); 16 17 var propertyType = typeof(TProperty); 18 19 if (QueryBuilder.IncludeInfos.Any(a => a.PropertyType.FullName == propertyType.FullName)) 20 { 21 throw new Exception($"屬性名稱:{result.SqlString} 不能重複使用Include方法."); 22 } 23 24 var type = propertyType; 25 26 if (type.IsArray) 27 { 28 type = type.GetElementType(); 29 } 30 else if (type.IsGenericType) 31 { 32 type = type.GenericTypeArguments[0]; 33 } 34 35 var queryBuilder = SqlBuilderFactory.CreateQueryBuilder(ado.DbOptions.DbType); 36 queryBuilder.EntityDbMapping = typeof(T).GetEntityDbMapping(); 37 queryBuilder.EntityDbMapping.Alias = "a"; 38 39 var includeInfo = new IncludeInfo(); 40 includeInfo.EntityDbMapping = type.GetEntityDbMapping(); 41 includeInfo.EntityDbMapping.Alias = "b"; 42 43 includeInfo.PropertyName = result.SqlString; 44 includeInfo.PropertyType = propertyType; 45 includeInfo.Type = type; 46 includeInfo.QueryBuilder = queryBuilder; 47 48 QueryBuilder.IncludeInfos.Add(includeInfo); 49 50 return new IncludeProvider<T, TProperty>(ado, QueryBuilder, includeInfo); 51 }
IncludeInfo 實體結構
1 using Fast.Framework.Abstract; 2 using Fast.Framework.Interfaces; 3 using System; 4 using System.Collections.Generic; 5 using System.Linq; 6 using System.Text; 7 using System.Threading.Tasks; 8 9 namespace Fast.Framework.Models 10 { 11 12 /// <summary> 13 /// 包括資訊 14 /// </summary> 15 public class IncludeInfo 16 { 17 /// <summary> 18 /// 屬性型別 19 /// </summary> 20 public Type PropertyType { get; set; } 21 22 /// <summary> 23 /// 型別 24 /// </summary> 25 public Type Type { get; set; } 26 27 /// <summary> 28 /// 屬性名稱 29 /// </summary> 30 public string PropertyName { get; set; } 31 32 /// <summary> 33 /// 實體資料庫對映 34 /// </summary> 35 public EntityDbMapping EntityDbMapping { get; set; } 36 37 /// <summary> 38 /// 條件列 39 /// </summary> 40 public string WhereColumn { get; set; } 41 42 /// <summary> 43 /// 查詢建造 44 /// </summary> 45 public QueryBuilder QueryBuilder { get; set; } 46 47 } 48 }
資料繫結核心類
1 using System; 2 using System.Reflection; 3 using System.Collections; 4 using System.Collections.Generic; 5 using System.Data.Common; 6 using System.Data; 7 using System.Linq; 8 using System.Text; 9 using System.Threading.Tasks; 10 using Fast.Framework.Abstract; 11 using Fast.Framework.Interfaces; 12 using Fast.Framework.Models; 13 14 namespace Fast.Framework.Extensions 15 { 16 17 /// <summary> 18 /// 查詢建造擴充套件類 19 /// </summary> 20 public static class QueryBuilderExtensions 21 { 22 23 private static readonly MethodInfo fristBuildMethod; 24 25 private static readonly MethodInfo listBuildMethod; 26 27 private static readonly MethodInfo ofTypeMethod; 28 29 private static readonly MethodInfo ofObjTypeMethod; 30 private static readonly MethodInfo ofObjTypeGenericMethod; 31 32 private static readonly MethodInfo toArrayMethod; 33 34 private static readonly MethodInfo toListMethod; 35 36 private static readonly MethodInfo toObjListMethod; 37 private static readonly MethodInfo toObjListGenericMethod; 38 39 /// <summary> 40 /// 構造方法 41 /// </summary> 42 static QueryBuilderExtensions() 43 { 44 fristBuildMethod = typeof(DbDataReaderExtensions).GetMethod("FristBuild", new Type[] { typeof(DbDataReader) }); 45 46 listBuildMethod = typeof(DbDataReaderExtensions).GetMethod("ListBuild", new Type[] { typeof(DbDataReader) }); 47 48 ofTypeMethod = typeof(Enumerable).GetMethod("OfType"); 49 50 ofObjTypeMethod = typeof(Enumerable).GetMethod("OfType"); 51 ofObjTypeGenericMethod = ofObjTypeMethod.MakeGenericMethod(typeof(object)); 52 53 toArrayMethod = typeof(Enumerable).GetMethod("ToArray"); 54 55 toListMethod = typeof(Enumerable).GetMethod("ToList"); 56 57 toObjListMethod = typeof(Enumerable).GetMethod("ToList"); 58 toObjListGenericMethod = toObjListMethod.MakeGenericMethod(typeof(object)); 59 } 60 61 /// <summary> 62 /// 初始化 63 /// </summary> 64 /// <param name="dbType">資料庫型別</param> 65 /// <param name="includeInfo">包括資訊</param> 66 /// <param name="isList">是否列表</param> 67 private static void Init(Models.DbType dbType, IncludeInfo includeInfo, bool isList) 68 { 69 var identifier = dbType.MappingIdentifier(); 70 var parameterSymbol = dbType.MappingParameterSymbol(); 71 72 //條件列 73 if (string.IsNullOrWhiteSpace(includeInfo.WhereColumn)) 74 { 75 var whereColumn = includeInfo.QueryBuilder.EntityDbMapping.ColumnsInfos.FirstOrDefault(f => f.IsPrimaryKey || f.ColumnName.ToUpper().EndsWith("ID")); 76 includeInfo.WhereColumn = whereColumn.ColumnName; 77 } 78 79 //排序列 80 var orderByColumn = includeInfo.EntityDbMapping.ColumnsInfos.FirstOrDefault(f => f.IsPrimaryKey || f.ColumnName.ToUpper().EndsWith("ID")); 81 if (orderByColumn != null) 82 { 83 if (includeInfo.QueryBuilder.OrderBy.Count == 0) 84 { 85 includeInfo.QueryBuilder.OrderBy.Add($"{identifier.Insert(1, includeInfo.EntityDbMapping.Alias)}.{identifier.Insert(1, orderByColumn.ColumnName)}"); 86 } 87 } 88 89 if (!isList) 90 { 91 includeInfo.QueryBuilder.Where.Add($"{identifier.Insert(1, includeInfo.EntityDbMapping.Alias)}.{identifier.Insert(1, includeInfo.WhereColumn)} = {parameterSymbol}{includeInfo.WhereColumn}"); 92 } 93 94 var joinInfo = new JoinInfo(); 95 joinInfo.IsInclude = true; 96 joinInfo.JoinType = JoinType.Inner; 97 joinInfo.EntityDbMapping = includeInfo.EntityDbMapping; 98 joinInfo.Where = $"{identifier.Insert(1, includeInfo.QueryBuilder.EntityDbMapping.Alias)}.{identifier.Insert(1, includeInfo.WhereColumn)} = {identifier.Insert(1, includeInfo.EntityDbMapping.Alias)}.{identifier.Insert(1, includeInfo.WhereColumn)}"; 99 100 includeInfo.QueryBuilder.Join.Add(joinInfo); 101 } 102 103 /// <summary> 104 /// Include資料繫結 105 /// </summary> 106 /// /// <param name="queryBuilder">查詢建造</param> 107 /// <param name="ado">Ado</param> 108 /// <param name="obj">物件</param> 109 /// <returns></returns> 110 public static void IncludeDataBind(this QueryBuilder queryBuilder, IAdo ado, object obj) 111 { 112 if (queryBuilder.IncludeInfos.Count > 0 && obj != null) 113 { 114 var type = obj.GetType(); 115 116 var isMultipleResult = false; 117 118 if (type.IsArray) 119 { 120 isMultipleResult = true; 121 type = type.GetElementType(); 122 } 123 else if (type.IsGenericType) 124 { 125 isMultipleResult = true; 126 type = type.GenericTypeArguments[0]; 127 } 128 129 foreach (var includeInfo in queryBuilder.IncludeInfos) 130 { 131 Init(ado.DbOptions.DbType, includeInfo, isMultipleResult); 132 133 var propertyInfo = type.GetProperty(includeInfo.PropertyName); 134 135 object data = null; 136 137 if (!isMultipleResult) 138 { 139 var parameterValue = type.GetProperty(includeInfo.WhereColumn).GetValue(obj); 140 includeInfo.QueryBuilder.DbParameters.Add(new DbParameterEx(includeInfo.WhereColumn, parameterValue)); 141 } 142 143 var sql = includeInfo.QueryBuilder.ToSqlString(); 144 var reader = ado.ExecuteReader(CommandType.Text, sql, ado.CreateParameter(includeInfo.QueryBuilder.DbParameters)); 145 146 var fristBuildGenericMethod = fristBuildMethod.MakeGenericMethod(includeInfo.Type); 147 148 var listBuildGenericMethod = listBuildMethod.MakeGenericMethod(includeInfo.Type); 149 150 var ofTypeGenericMethod = ofTypeMethod.MakeGenericMethod(includeInfo.Type); 151 152 var toArrayGenericMethod = toArrayMethod.MakeGenericMethod(includeInfo.Type); 153 154 var toListGenericMethod = toListMethod.MakeGenericMethod(includeInfo.Type); 155 156 if (isMultipleResult) 157 { 158 data = listBuildGenericMethod.Invoke(null, new object[] { reader }); 159 160 data = ofObjTypeGenericMethod.Invoke(null, new object[] { data }); 161 162 data = toObjListGenericMethod.Invoke(null, new object[] { data }); 163 164 var list = data as List<object>; 165 166 if (list.Any()) 167 { 168 var whereColumnProInfo = list.FirstOrDefault()?.GetType().GetProperty(includeInfo.WhereColumn); 169 if (whereColumnProInfo != null) 170 { 171 foreach (var item in obj as IList) 172 { 173 var parameterValue = type.GetProperty(includeInfo.WhereColumn).GetValue(item); 174 175 object value = null; 176 177 if (includeInfo.PropertyType.IsArray || includeInfo.PropertyType.IsGenericType) 178 { 179 value = list.Where(w => Convert.ToString(whereColumnProInfo.GetValue(w)) == Convert.ToString(parameterValue)); 180 181 value = ofTypeGenericMethod.Invoke(null, new object[] { value }); 182 183 if (includeInfo.PropertyType.IsArray) 184 { 185 value = toArrayGenericMethod.Invoke(null, new object[] { value }); 186 } 187 else if (includeInfo.PropertyType.IsGenericType) 188 { 189 value = toListGenericMethod.Invoke(null, new object[] { value }); 190 } 191 } 192 else 193 { 194 value = list.FirstOrDefault(w => Convert.ToString(whereColumnProInfo.GetValue(w)) == Convert.ToString(parameterValue)).ChanageType(includeInfo.PropertyType); 195 } 196 197 propertyInfo.SetValue(item, value); 198 } 199 } 200 } 201 } 202 else 203 { 204 if (includeInfo.PropertyType.IsArray || includeInfo.PropertyType.IsGenericType) 205 { 206 data = listBuildGenericMethod.Invoke(null, new object[] { reader }); 207 208 if (includeInfo.PropertyType.IsArray) 209 { 210 data = toArrayGenericMethod.Invoke(null, new object[] { data }); 211 } 212 } 213 else 214 { 215 data = fristBuildGenericMethod.Invoke(null, new object[] { reader }); 216 } 217 propertyInfo.SetValue(obj, data); 218 } 219 220 if (includeInfo.QueryBuilder.IncludeInfos.Count > 0) 221 { 222 includeInfo.QueryBuilder.IncludeDataBind(ado, data); 223 } 224 225 } 226 } 227 } 228 229 } 230 }