123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Reflection;
- using System.Text;
- using System.Threading.Tasks;
- namespace Ropin.Inspection.Repository
- {
- internal static class LinqExtensions
- {
- private static PropertyInfo GetPropertyInfo(Type objType, string name)
- {
- var properties = objType.GetProperties();
- var matchedProperty = properties.FirstOrDefault(p => p.Name == name);
- if (matchedProperty == null)
- {
- throw new ArgumentException("name");
- }
- return matchedProperty;
- }
- private static LambdaExpression GetOrderExpression(Type objType, PropertyInfo pi)
- {
- var paramExpr = Expression.Parameter(objType);
- var propAccess = Expression.PropertyOrField(paramExpr, pi.Name);
- var expr = Expression.Lambda(propAccess, paramExpr);
- return expr;
- }
- /// <summary>
- /// 多个OrderBy用逗号隔开,属性前面带-号表示反序排序,exp:"name,-createtime"
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IEnumerable<T> OrderByBatch<T>(this IEnumerable<T> query, string name)
- {
- var index = 0;
- var a = name.Split(',');
- foreach (var item in a)
- {
- var m = index++ > 0 ? "ThenBy" : "OrderBy";
- if (item.StartsWith("-"))
- {
- m += "Descending";
- name = item.Substring(1);
- }
- else
- {
- name = item;
- }
- name = name.Trim();
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var method = typeof(Enumerable).GetMethods().FirstOrDefault(mt => mt.Name == m && mt.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- query = (IEnumerable<T>)genericMethod.Invoke(null, new object[] { query, expr.Compile() });
- }
- return query;
- }
- /// <summary>
- /// 多个OrderBy用逗号隔开,属性前面带-号表示反序排序,exp:"name,-createtime"
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IQueryable<T> OrderByBatch<T>(this IQueryable<T> query, string name)
- {
- var index = 0;
- var a = name.Split(',');
- foreach (var item in a)
- {
- var m = index++ > 0 ? "ThenBy" : "OrderBy";
- if (item.StartsWith("-"))
- {
- m += "Descending";
- name = item.Substring(1);
- }
- else
- {
- name = item;
- }
- name = name.Trim();
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var method = typeof(Queryable).GetMethods().FirstOrDefault(mt => mt.Name == m && mt.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- query = (IQueryable<T>)genericMethod.Invoke(null, new object[] { query, expr });
- }
- return query;
- }
- /// <summary>
- /// 正序排序单个
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IQueryable<T> OrderBy<T>(this IQueryable<T> query, string name)
- {
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var method = typeof(Queryable).GetMethods().FirstOrDefault(m => m.Name == "OrderBy" && m.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- return (IQueryable<T>)genericMethod.Invoke(null, new object[] { query, expr });
- }
- /// <summary>
- /// 正序排序单个(非首个)
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IQueryable<T> ThenBy<T>(this IQueryable<T> query, string name)
- {
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var method = typeof(Queryable).GetMethods().FirstOrDefault(m => m.Name == "ThenBy" && m.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- return (IQueryable<T>)genericMethod.Invoke(null, new object[] { query, expr });
- }
- /// <summary>
- /// 反序排序单个
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IQueryable<T> OrderByDescending<T>(this IQueryable<T> query, string name)
- {
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var metMethods = typeof(Queryable).GetMethods();
- var method = metMethods.FirstOrDefault(m => m.Name == "OrderByDescending" && m.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- return (IQueryable<T>)genericMethod.Invoke(null, new object[] { query, expr });
- }
- /// <summary>
- /// 反序排序单个(非首个)
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="query"></param>
- /// <param name="name"></param>
- /// <returns></returns>
- public static IQueryable<T> ThenByDescending<T>(this IQueryable<T> query, string name)
- {
- var propInfo = GetPropertyInfo(typeof(T), name);
- var expr = GetOrderExpression(typeof(T), propInfo);
- var metMethods = typeof(Queryable).GetMethods();
- var method = metMethods.FirstOrDefault(m => m.Name == "ThenByDescending" && m.GetParameters().Length == 2);
- var genericMethod = method.MakeGenericMethod(typeof(T), propInfo.PropertyType);
- return (IQueryable<T>)genericMethod.Invoke(null, new object[] { query, expr });
- }
- }
- }
|