[英]C# EF6 conditional property selection?
假設我有代碼優先模型:
public class FooBar
{
[Key]
public int Id {get;set;}
[MaxLength(254)]
public string Title {get;set;}
public string Description {get;set;}
}
以及檢索行的一些數據子集的方法:
public IQueryable<FooBar> GetDataQuery(bool includeTitle, bool includeDescription)
{
var query = ctx.FooBars.AsNoTracking().Where(Id > 123);
//how to inlcude/exclude???
return query;
}
問題是如何在不對匿名類型進行硬編碼的情況下使用特定字段構建查詢? 基本上,我想告訴SQL查詢構建器使用指定的字段構建查詢,而不在客戶端上進行后期過濾。 因此,如果我排除描述 - 它將不會通過電匯發送。
此外,有這樣的經驗:
public IQueryable<FooBar> GetDataQuery(bool includeTitle, bool includeDescription)
{
var query = ctx.FooBars.AsNoTracking().Where(Id > 123);
query = query.Select(x=> new
{
Id = x.Id
Title = includeTitle ? x.Title : null,
Description = includeDescription ? x.Description : null,
})
.MapBackToFooBarsSomehow();//this will fail, I know, do not want to write boilerplate to hack this out, just imagine return type will be correctly retrieved
return query;
}
但是這將通過有線的includeTitle , includeDescription屬性作為EXEC的 SQL參數發送,並且在大多數情況下查詢與沒有這種混亂的簡單非條件匿名查詢相比效率低 - 但是編寫匿名結構的每個可能的排列都不是一種選擇。
PS :實際上有大量的“包含/排除”屬性,我只是為了簡單而提出了兩個。
更新:
受@reckface答案的啟發,我為那些希望在查詢結束時實現流暢的執行和映射到實體的人編寫了擴展:
public static class CustomSqlMapperExtension
{
public sealed class SpecBatch<T>
{
internal readonly List<Expression<Func<T, object>>> Items = new List<Expression<Func<T, object>>>();
internal SpecBatch()
{
}
public SpecBatch<T> Property(Expression<Func<T, object>> selector, bool include = true)
{
if (include)
{
Items.Add(selector);
}
return this;
}
}
public static List<T> WithCustom<T>(this IQueryable<T> source, Action<SpecBatch<T>> configurator)
{
if (source == null)
return null;
var batch = new SpecBatch<T>();
configurator(batch);
if (!batch.Items.Any())
throw new ArgumentException("Nothing selected from query properties", nameof(configurator));
LambdaExpression lambda = CreateSelector(batch);
var rawQuery = source.Provider.CreateQuery(
Expression.Call(
typeof(Queryable),
nameof(Queryable.Select),
new[]
{
source.ElementType,
lambda.Body.Type
},
source.Expression,
Expression.Quote(lambda))
);
return rawQuery.ToListAsync().Result.ForceCast<T>().ToList();
}
private static IEnumerable<T> ForceCast<T>(this IEnumerable<object> enumer)
{
return enumer.Select(x=> Activator.CreateInstance(typeof(T)).ShallowAssign(x)).Cast<T>();
}
private static object ShallowAssign(this object target, object source)
{
if (target == null || source == null)
throw new ArgumentNullException();
var type = target.GetType();
var data = source.GetType().GetProperties()
.Select(e => new
{
e.Name,
Value = e.GetValue(source)
});
foreach (var property in data)
{
type.GetProperty(property.Name).SetValue(target, property.Value);
}
return target;
}
private static LambdaExpression CreateSelector<T>(SpecBatch<T> batch)
{
var input = "new(" + string.Join(", ", batch.Items.Select(GetMemberName<T>)) + ")";
return System.Linq.Dynamic.DynamicExpression.ParseLambda(typeof(T), null, input);
}
private static string GetMemberName<T>(Expression<Func<T, object>> expr)
{
var body = expr.Body;
if (body.NodeType == ExpressionType.Convert)
{
body = ((UnaryExpression) body).Operand;
}
var memberExpr = body as MemberExpression;
var propInfo = memberExpr.Member as PropertyInfo;
return propInfo.Name;
}
}
用法:
public class Topic
{
public long Id { get; set; }
public string Title { get; set; }
public string Body { get; set; }
public string Author { get; set; }
public byte[] Logo { get; set; }
public bool IsDeleted { get; set; }
}
public class MyContext : DbContext
{
public DbSet<Topic> Topics { get; set; }
}
class Program
{
static void Main(string[] args)
{
using (var ctx = new MyContext())
{
ctx.Database.Log = Console.WriteLine;
var query = (ctx.Topics ?? Enumerable.Empty<Topic>()).AsQueryable();
query = query.Where(x => x.Title != null);
var result = query.WithCustom(
cfg => cfg //include whitelist config
.Property(x => x.Author, true) //include
.Property(x => x.Title, false) //exclude
.Property(x=> x.Id, true)); //include
}
}
}
重要的是,在您明確附加它們之前,不能在EF中使用這些實體。
據我所知,在EF中沒有干凈的方法。 你可以使用一些各種丑陋的解決方法,下面是一個。 只有當你不打算更新\\ attach \\ delete返回的實體時,它才會起作用,我認為這個用例很好。
假設我們只想包含屬性“ID”和“代碼”。 我們需要構造這種形式的表達式:
fooBarsQuery.Select(x => new FooBar {ID = x.ID, Code = x.Code))
我們可以像這樣手動完成:
public static IQueryable<T> IncludeOnly<T>(this IQueryable<T> query, params string[] properties) {
var arg = Expression.Parameter(typeof(T), "x");
var bindings = new List<MemberBinding>();
foreach (var propName in properties) {
var prop = typeof(T).GetProperty(propName);
bindings.Add(Expression.Bind(prop, Expression.Property(arg, prop)));
}
// our select, x => new T {Prop1 = x.Prop1, Prop2 = x.Prop2 ...}
var select = Expression.Lambda<Func<T, T>>(Expression.MemberInit(Expression.New(typeof(T)), bindings), arg);
return query.Select(select);
}
但如果我們真的嘗試:
// some test entity I use
var t = ctx.Errors.IncludeOnly("ErrorID", "ErrorCode", "Duration").Take(10).ToList();
它將失敗,但例外
實體或復雜類型...不能在LINQ to Entities查詢中構造
因此,如果SomeType
是映射實體的類型,則new SomeType
在Select
是非法的。
但是如果我們有一個從實體繼承的類型並使用它呢?
public class SomeTypeProxy : SomeType {}
好吧,那就行了。 所以我們需要在某處獲得這樣的代理類型。 使用內置工具在運行時生成它很容易,因為我們所需要的只是從某種類型繼承而且都是。
考慮到這一點,我們的方法變為:
static class Extensions {
private static ModuleBuilder _moduleBuilder;
private static readonly Dictionary<Type, Type> _proxies = new Dictionary<Type, Type>();
static Type GetProxyType<T>() {
lock (typeof(Extensions)) {
if (_proxies.ContainsKey(typeof(T)))
return _proxies[typeof(T)];
if (_moduleBuilder == null) {
var asmBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(
new AssemblyName("ExcludeProxies"), AssemblyBuilderAccess.Run);
_moduleBuilder = asmBuilder.DefineDynamicModule(
asmBuilder.GetName().Name, false);
}
// Create a proxy type
TypeBuilder typeBuilder = _moduleBuilder.DefineType(typeof(T).Name + "Proxy",
TypeAttributes.Public |
TypeAttributes.Class,
typeof(T));
var type = typeBuilder.CreateType();
// cache it
_proxies.Add(typeof(T), type);
return type;
}
}
public static IQueryable<T> IncludeOnly<T>(this IQueryable<T> query, params string[] properties) {
var arg = Expression.Parameter(typeof(T), "x");
var bindings = new List<MemberBinding>();
foreach (var propName in properties) {
var prop = typeof(T).GetProperty(propName);
bindings.Add(Expression.Bind(prop, Expression.Property(arg, prop)));
}
// modified select, (T x) => new TProxy {Prop1 = x.Prop1, Prop2 = x.Prop2 ...}
var select = Expression.Lambda<Func<T, T>>(Expression.MemberInit(Expression.New(GetProxyType<T>()), bindings), arg);
return query.Select(select);
}
}
現在它工作正常並生成只包含字段的select sql查詢。 它確實返回了一個代理類型列表,但這不是問題,因為代理類型繼承自您的查詢類型。 按照我之前的說法 - 你不能附加\\ update \\從上下文中刪除它。
當然你也可以修改這個方法來排除,接受屬性表達式而不是純字符串等等,這只是想法證明代碼。
我非常成功地使用了System.Linq.Dynamic 。 您可以按以下格式將字符串作為select語句傳遞: .Select("new(Title, Description)")
所以你的例子將成為:
// ensure you import the System.Linq.Dynamic namespace
public IQueryable<FooBar> GetDataQuery(bool includeTitle, bool includeDescription)
{
// build a list of columns, at least 1 must be selected, so maybe include an Id
var columns = new List<string>(){nameof(FooBar.Id)};
if (includeTitle)
columns.Add(nameof(FooBar.Title));
if (includeDescription)
columns.Add(nameof(FooBar.Description));
// join said columns
var select = $"new({string.Join(", ", columns)})";
var query = ctx.FooBars.AsQueryable()
.Where(f => f.Id > 240)
.Select(select)
.OfType<FooBar>();
return query;
}
編輯
變成OfType()可能在這里不起作用。 如果是這樣的話,這是一個窮人的擴展方法:
// not ideal, but it fits your constraints
var query = ctx.FooBars.AsQueryable()
.Where(f => f.Id > 240)
.Select(select)
.ToListAsync().Result
.Select(r => new FooBar().Fill(r));
public static T Fill<T>(this T item, object element)
{
var type = typeof(T);
var data = element.GetType().GetProperties()
.Select(e => new
{
e.Name,
Value = e.GetValue(element)
});
foreach (var property in data)
{
type.GetProperty(property.Name).SetValue(item, property.Value);
}
return item;
}
更新
但等等還有更多!
var query = ctx.FooBars
.Where(f => f.Id > 240)
.Select(select)
.ToJson() // using Newtonsoft.JSON, I know, I know, awful.
.FromJson<IEnumerable<FooBar>>()
.AsQueryable(); // this is no longer valid or necessary
return query;
public static T FromJson<T>(this string json)
{
var serializer = new JsonSerializer();
using (var sr = new StringReader(json))
using (var jr = new JsonTextReader(sr))
{
var result = serializer.Deserialize<T>(jr);
return result;
}
}
public static string ToJson(this object data)
{
if (data == null)
return null;
var json = JsonConvert.SerializeObject(data, Newtonsoft.Json.Formatting.Indented);
return json;
}
結果
使用布爾類型標記包含字段是不可持續的,尤其是當您有一長串字段時。 更好的方法是為過濾器設置可選參數,並在將值添加到查詢之前檢查該值。 應謹慎選擇可選參數的值。
例如,給出以下模型
public class FooBar
{
[Key]
public int Id {get;set;}
[MaxLength(254)]
public string Title {get;set;}
public string Description {get;set;}
}
知道Title字段不能為空。 我可以構建我的查詢
public IQueryable<FooBar> GetDataQuery(string title = "")
{
var query = ctx.FooBars.AsNoTracking().Where(Id > 123);
if(!string.isnullorempty(title)
{
query = query.where(x=>x.title = title)
}
return query;
}
我知道這里選擇可選參數可能很棘手。 我希望這有幫助
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.