lambda表达式转换sql

帅比萌擦擦* 提交于 2020-01-10 08:40:23

 

这是我在博客园的第一遍文章,想分享下lambda表达式转换sql。

喜欢EF的便捷与优雅,不喜欢生成的一坨sql。(PS:公司封装了一套访问数据库的方法,所以不确定是不是EF的问题,反正就是一坨密密麻麻的的sql,我有点点处女座小纠结,虽然我是天蝎座)

好了,废话少说。

  1     public class LambdaToSqlHelper
  2     {
  3         /// <summary>
  4         /// NodeType枚举
  5         /// </summary>
  6         private enum EnumNodeType
  7         {
  8             /// <summary>
  9             /// 二元运算符
 10             /// </summary>
 11             [Description("二元运算符")]
 12             BinaryOperator = 1,
 13 
 14             /// <summary>
 15             /// 一元运算符
 16             /// </summary>
 17             [Description("一元运算符")]
 18             UndryOperator = 2,
 19 
 20             /// <summary>
 21             /// 常量表达式
 22             /// </summary>
 23             [Description("常量表达式")]
 24             Constant = 3,
 25 
 26             /// <summary>
 27             /// 成员(变量)
 28             /// </summary>
 29             [Description("成员(变量)")]
 30             MemberAccess = 4,
 31 
 32             /// <summary>
 33             /// 函数
 34             /// </summary>
 35             [Description("函数")]
 36             Call = 5,
 37 
 38             /// <summary>
 39             /// 未知
 40             /// </summary>
 41             [Description("未知")]
 42             Unknown = -99,
 43 
 44             /// <summary>
 45             /// 不支持
 46             /// </summary>
 47             [Description("不支持")]
 48             NotSupported = -98
 49         }
 50 
 51         /// <summary>
 52         /// 判断表达式类型
 53         /// </summary>
 54         /// <param name="exp">lambda表达式</param>
 55         /// <returns></returns>
 56         private static EnumNodeType CheckExpressionType(Expression exp)
 57         {
 58             switch (exp.NodeType)
 59             {
 60                 case ExpressionType.AndAlso:
 61                 case ExpressionType.OrElse:
 62                 case ExpressionType.Equal:
 63                 case ExpressionType.GreaterThanOrEqual:
 64                 case ExpressionType.LessThanOrEqual:
 65                 case ExpressionType.GreaterThan:
 66                 case ExpressionType.LessThan:
 67                 case ExpressionType.NotEqual:
 68                     return EnumNodeType.BinaryOperator;
 69                 case ExpressionType.Constant:
 70                     return EnumNodeType.Constant;
 71                 case ExpressionType.MemberAccess:
 72                     return EnumNodeType.MemberAccess;
 73                 case ExpressionType.Call:
 74                     return EnumNodeType.Call;
 75                 case ExpressionType.Not:
 76                 case ExpressionType.Convert:
 77                     return EnumNodeType.UndryOperator;
 78                 default:
 79                     return EnumNodeType.Unknown;
 80             }
 81         }
 82 
 83         /// <summary>
 84         /// 表达式类型转换
 85         /// </summary>
 86         /// <param name="type"></param>
 87         /// <returns></returns>
 88         private static string ExpressionTypeCast(ExpressionType type)
 89         {
 90             switch (type)
 91             {
 92                 case ExpressionType.And:
 93                 case ExpressionType.AndAlso:
 94                     return " and ";
 95                 case ExpressionType.Equal:
 96                     return " = ";
 97                 case ExpressionType.GreaterThan:
 98                     return " > ";
 99                 case ExpressionType.GreaterThanOrEqual:
100                     return " >= ";
101                 case ExpressionType.LessThan:
102                     return " < ";
103                 case ExpressionType.LessThanOrEqual:
104                     return " <= ";
105                 case ExpressionType.NotEqual:
106                     return " <> ";
107                 case ExpressionType.Or:
108                 case ExpressionType.OrElse:
109                     return " or ";
110                 case ExpressionType.Add:
111                 case ExpressionType.AddChecked:
112                     return " + ";
113                 case ExpressionType.Subtract:
114                 case ExpressionType.SubtractChecked:
115                     return " - ";
116                 case ExpressionType.Divide:
117                     return " / ";
118                 case ExpressionType.Multiply:
119                 case ExpressionType.MultiplyChecked:
120                     return " * ";
121                 default:
122                     return null;
123             }
124         }
125 
126         private static string BinarExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
127         {
128             BinaryExpression be = exp as BinaryExpression;
129             Expression left = be.Left;
130             Expression right = be.Right;
131             ExpressionType type = be.NodeType;
132             string sb = "(";
133             //先处理左边
134             sb += ExpressionRouter(left, listSqlParaModel);
135             sb += ExpressionTypeCast(type);
136             //再处理右边
137             string sbTmp = ExpressionRouter(right, listSqlParaModel);
138             if (sbTmp == "null")
139             {
140                 if (sb.EndsWith(" = "))
141                     sb = sb.Substring(0, sb.Length - 2) + " is null";
142                 else if (sb.EndsWith(" <> "))
143                     sb = sb.Substring(0, sb.Length - 2) + " is not null";
144             }
145             else
146                 sb += sbTmp;
147             return sb += ")";
148         }
149 
150         private static string ConstantExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
151         {
152             ConstantExpression ce = exp as ConstantExpression;
153             if (ce.Value == null)
154             {
155                 return "null";
156             }
157             else if (ce.Value is ValueType)
158             {
159                 GetSqlParaModel(listSqlParaModel, GetValueType(ce.Value));
160                 return "@para" + listSqlParaModel.Count;
161             }
162             else if (ce.Value is string || ce.Value is DateTime || ce.Value is char)
163             {
164                 GetSqlParaModel(listSqlParaModel, GetValueType(ce.Value));
165                 return "@para" + listSqlParaModel.Count;
166             }
167             return "";
168         }
169 
170         private static string LambdaExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
171         {
172             LambdaExpression le = exp as LambdaExpression;
173             return ExpressionRouter(le.Body, listSqlParaModel);
174         }
175 
176         private static string MemberExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
177         {
178             if (!exp.ToString().StartsWith("value"))
179             {
180                 MemberExpression me = exp as MemberExpression;
181                 if (me.Member.Name == "Now")
182                 {
183                     GetSqlParaModel(listSqlParaModel, DateTime.Now);
184                     return "@para" + listSqlParaModel.Count;
185                 }
186                 return me.Member.Name;
187             }
188             else
189             {
190                 var result = Expression.Lambda(exp).Compile().DynamicInvoke();
191                 if (result == null)
192                 {
193                     return "null";
194                 }
195                 else if (result is ValueType)
196                 {
197                     GetSqlParaModel(listSqlParaModel, GetValueType(result));
198                     return "@para" + listSqlParaModel.Count;
199                 }
200                 else if (result is string || result is DateTime || result is char)
201                 {
202                     GetSqlParaModel(listSqlParaModel, GetValueType(result));
203                     return "@para" + listSqlParaModel.Count;
204                 }
205                 else if (result is int[])
206                 {
207                     var rl = result as int[];
208                     StringBuilder sbTmp = new StringBuilder();
209                     foreach (var r in rl)
210                     {
211                         GetSqlParaModel(listSqlParaModel, r.ToString().ToInt32());
212                         sbTmp.Append("@para" + listSqlParaModel.Count + ",");
213                     }
214                     return sbTmp.ToString().Substring(0, sbTmp.ToString().Length - 1);
215                 }
216                 else if (result is string[])
217                 {
218                     var rl = result as string[];
219                     StringBuilder sbTmp = new StringBuilder();
220                     foreach (var r in rl)
221                     {
222                         GetSqlParaModel(listSqlParaModel, r.ToString());
223                         sbTmp.Append("@para" + listSqlParaModel.Count + ",");
224                     }
225                     return sbTmp.ToString().Substring(0, sbTmp.ToString().Length - 1);
226                 }                
227             }
228             return "";
229         }
230 
231         private static string MethodCallExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
232         {
233             MethodCallExpression mce = exp as MethodCallExpression;
234             if (mce.Method.Name == "Contains")
235             {
236                 if (mce.Object == null)
237                 {
238                     return string.Format("{0} in ({1})", ExpressionRouter(mce.Arguments[1], listSqlParaModel), ExpressionRouter(mce.Arguments[0], listSqlParaModel));
239                 }
240                 else
241                 {
242                     if (mce.Object.NodeType == ExpressionType.MemberAccess)
243                     {
244                         //w => w.name.Contains("1")
245                         var _name = ExpressionRouter(mce.Object, listSqlParaModel);
246                         var _value = ExpressionRouter(mce.Arguments[0], listSqlParaModel);
247                         var index = _value.RetainNumber().ToInt32() - 1;
248                         listSqlParaModel[index].value = "%{0}%".FormatWith(listSqlParaModel[index].value);
249                         return string.Format("{0} like {1}", _name, _value);
250                     }
251                 }
252             }
253             else if (mce.Method.Name == "OrderBy")
254             {
255                 return string.Format("{0} asc", ExpressionRouter(mce.Arguments[1], listSqlParaModel));
256             }
257             else if (mce.Method.Name == "OrderByDescending")
258             {
259                 return string.Format("{0} desc", ExpressionRouter(mce.Arguments[1], listSqlParaModel));
260             }
261             else if (mce.Method.Name == "ThenBy")
262             {
263                 return string.Format("{0},{1} asc", MethodCallExpressionProvider(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel));
264             }
265             else if (mce.Method.Name == "ThenByDescending")
266             {
267                 return string.Format("{0},{1} desc", MethodCallExpressionProvider(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel));
268             }
269             else if (mce.Method.Name == "Like")
270             {
271                 return string.Format("({0} like {1})", ExpressionRouter(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel).Replace("'", ""));
272             }
273             else if (mce.Method.Name == "NotLike")
274             {
275                 return string.Format("({0} not like '%{1}%')", ExpressionRouter(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel).Replace("'", ""));
276             }
277             else if (mce.Method.Name == "In")
278             {
279                 return string.Format("{0} in ({1})", ExpressionRouter(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel));
280             }
281             else if (mce.Method.Name == "NotIn")
282             {
283                 return string.Format("{0} not in ({1})", ExpressionRouter(mce.Arguments[0], listSqlParaModel), ExpressionRouter(mce.Arguments[1], listSqlParaModel));
284             }
285             return "";
286         }
287 
288         private static string NewArrayExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
289         {
290             NewArrayExpression ae = exp as NewArrayExpression;
291             StringBuilder sbTmp = new StringBuilder();
292             foreach (Expression ex in ae.Expressions)
293             {
294                 sbTmp.Append(ExpressionRouter(ex, listSqlParaModel));
295                 sbTmp.Append(",");
296             }
297             return sbTmp.ToString(0, sbTmp.Length - 1);
298         }
299 
300         private static string ParameterExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
301         {
302             ParameterExpression pe = exp as ParameterExpression;
303             return pe.Type.Name;
304         }
305 
306         private static string UnaryExpressionProvider(Expression exp, List<SqlParaModel> listSqlParaModel)
307         {
308             UnaryExpression ue = exp as UnaryExpression;
309             var result = ExpressionRouter(ue.Operand, listSqlParaModel);
310             ExpressionType type = exp.NodeType;
311             if (type == ExpressionType.Not)
312             {
313                 if (result.Contains(" in "))
314                 {
315                     result = result.Replace(" in ", " not in ");
316                 }
317                 if (result.Contains(" like "))
318                 {
319                     result = result.Replace(" like ", " not like ");
320                 }
321             }
322             return result;
323         }
324 
325         /// <summary>
326         /// 路由计算
327         /// </summary>
328         /// <param name="exp"></param>
329         /// <param name="listSqlParaModel"></param>
330         /// <returns></returns>
331         private static string ExpressionRouter(Expression exp, List<SqlParaModel> listSqlParaModel)
332         {
333             var nodeType = exp.NodeType;
334             if (exp is BinaryExpression)    //表示具有二进制运算符的表达式
335             {
336                 return BinarExpressionProvider(exp, listSqlParaModel);
337             }
338             else if (exp is ConstantExpression) //表示具有常数值的表达式
339             {
340                 return ConstantExpressionProvider(exp, listSqlParaModel);
341             }
342             else if (exp is LambdaExpression)   //介绍 lambda 表达式。 它捕获一个类似于 .NET 方法主体的代码块
343             {
344                 return LambdaExpressionProvider(exp, listSqlParaModel);
345             }
346             else if (exp is MemberExpression)   //表示访问字段或属性
347             {
348                 return MemberExpressionProvider(exp, listSqlParaModel);
349             }
350             else if (exp is MethodCallExpression)   //表示对静态方法或实例方法的调用
351             {
352                 return MethodCallExpressionProvider(exp, listSqlParaModel);
353             }
354             else if (exp is NewArrayExpression) //表示创建一个新数组,并可能初始化该新数组的元素
355             {
356                 return NewArrayExpressionProvider(exp, listSqlParaModel);
357             }
358             else if (exp is ParameterExpression)    //表示一个命名的参数表达式。
359             {
360                 return ParameterExpressionProvider(exp, listSqlParaModel);
361             }
362             else if (exp is UnaryExpression)    //表示具有一元运算符的表达式
363             {
364                 return UnaryExpressionProvider(exp, listSqlParaModel);
365             }
366             return null;
367         }
368 
369         /// <summary>
370         /// 值类型转换
371         /// </summary>
372         /// <param name="_value"></param>
373         /// <returns></returns>
374         private static object GetValueType(object _value)
375         {
376             var _type = _value.GetType().Name;
377             switch (_type)
378             {
379                 case "Decimal ": return _value.ToDecimal();
380                 case "Int32": return _value.ToInt32();
381                 case "DateTime": return _value.ToDateTime();
382                 case "String": return _value.ToString();
383                 case "Char":return _value.ToChar();
384                 case "Boolean":return _value.ToBoolean();
385                 default: return _value;
386             }
387         }
388 
389         /// <summary>
390         /// sql参数
391         /// </summary>
392         /// <param name="listSqlParaModel"></param>
393         /// <param name="val"></param>
394         private static void GetSqlParaModel(List<SqlParaModel> listSqlParaModel, object val)
395         {
396             SqlParaModel p = new SqlParaModel();
397             p.name = "para" + (listSqlParaModel.Count + 1);
398             p.value = val;
399             listSqlParaModel.Add(p);
400         }
401 
402         /// <summary>
403         /// lambda表达式转换sql
404         /// </summary>
405         /// <typeparam name="T"></typeparam>
406         /// <param name="where"></param>
407         /// <param name="listSqlParaModel"></param>
408         /// <returns></returns>
409         public static string GetWhereSql<T>(Expression<Func<T, bool>> where, List<SqlParaModel> listSqlParaModel) where T : class
410         {
411             string result = string.Empty;
412             if (where != null)
413             {
414                 Expression exp = where.Body as Expression;
415                 result = ExpressionRouter(exp, listSqlParaModel);
416             }
417             if (result != string.Empty)
418             {
419                 result = " where " + result;
420             }
421             return result;
422         }
423 
424         /// <summary>
425         /// lambda表达式转换sql
426         /// </summary>
427         /// <typeparam name="T"></typeparam>
428         /// <param name="orderBy"></param>
429         /// <returns></returns>
430         public static string GetOrderBySql<T>(Expression<Func<IQueryable<T>, IOrderedQueryable<T>>> orderBy) where T : class
431         {
432             string result = string.Empty;
433             if (orderBy != null && orderBy.Body is MethodCallExpression)
434             {
435                 MethodCallExpression exp = orderBy.Body as MethodCallExpression;
436                 List<SqlParaModel> listSqlParaModel = new List<SqlParaModel>();
437                 result = MethodCallExpressionProvider(exp, listSqlParaModel);
438             }
439             if (result != string.Empty)
440             {
441                 result = " order by " + result;
442             }
443             return result;
444         }
445 
446         /// <summary>
447         /// lambda表达式转换sql
448         /// </summary>
449         /// <typeparam name="T"></typeparam>
450         /// <param name="fields"></param>
451         /// <returns></returns>
452         public static string GetQueryField<T>(Expression<Func<T, object>> fields)
453         {
454             StringBuilder sbSelectFields = new StringBuilder();
455             if (fields.Body is NewExpression)
456             {
457                 NewExpression ne = fields.Body as NewExpression;
458                 for (var i = 0; i < ne.Members.Count; i++)
459                 {
460                     sbSelectFields.Append(ne.Members[i].Name + ",");
461                 }
462             }
463             else if (fields.Body is ParameterExpression)
464             {
465                 sbSelectFields.Append("*");
466             }
467             else
468             {
469                 sbSelectFields.Append("*");
470             }
471             if (sbSelectFields.Length > 1)
472             {
473                 sbSelectFields = sbSelectFields.Remove(sbSelectFields.Length - 1, 1);
474             }
475             return sbSelectFields.ToString();
476         }
477 
478     }
View Code

SqlParaModel如下:

 1     public class SqlParaModel
 2     {
 3         /// <summary>
 4         /// 
 5         /// </summary>
 6         public string name { set; get; }
 7 
 8         /// <summary>
 9         /// 
10         /// </summary>
11         public object value { set; get; }
12     }

demo:

 1     class MyClass
 2     {
 3         public string id;
 4         public string name;
 5         public string desc;
 6         public decimal price;
 7         public int stock;
 8         public bool isShow;
 9         public DateTime createTime;
10     }
 1     class Program
 2     {
 3 
 4         static void Main(string[] args)
 5         {
 6             //Expression<Func<MyClass, bool>> where = w => w.id == "123456";
 7             Expression<Func<MyClass, bool>> where = w => w.id.Contains("1");
 8             List<SqlParaModel> listSqlParaModel = new List<SqlParaModel>();
 9             var sql = LambdaToSqlHelper.GetWhereSql(where, listSqlParaModel);
10         }
11 
12     }

参考:

http://www.cnblogs.com/kakura/p/6108950.html
http://www.cnblogs.com/zhyue/p/5690807.html

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!