.NET中l(wèi)ambda表達式合并問題及解決方法
事情的起因是公司一個小伙子問了我個問題 “海哥,來幫我看下這段代碼怎么不行”
Func<Report,bool> nameFilter = x=>x.Name == "test"; DbContext.Report.Where(x=>x.State==1 && nameFilter(x));
我一看,好家伙,這么騷的代碼都能讓你想出來,正常情況下用Linq To Object是可以這么操作的,但是EF的IQueryable查詢是不能這么操作的。
Linq To Object是直接執(zhí)行表達式,他就是個委托方法,里面嵌套多少層委托和方法都是能直接執(zhí)行的IQueryable并不會執(zhí)行表達式和方法,是把表達式轉(zhuǎn)換為對應(yīng)的Sql語句來執(zhí)行,解析到nameFilter的時候他就懵逼了,這是啥玩意兒啊,sql里面沒有這種東西啊,他就轉(zhuǎn)換不了了。
小伙子知道后明細(xì)很失望,那不能啊,也不是我想顯擺我的技術(shù),就是想讓小伙子能繼續(xù)他的騷操作,給他來點??怂箍萍寂c狠活。
解決方案:
//表達式 Func<Report,bool> nameFilter = x=>x.Name == "test"; Func<Report,bool> stateFilter = x=>x.State==1; //合并為 Func<Report,bool> whereFilter = x=>x.Name == "test" && x.State==1; //調(diào)用 DbContext.Report.Where(whereFilter);
完美解決
那怎么合并,當(dāng)然得自己構(gòu)造一個新的表達式,構(gòu)造表達式需要用到Expression類,如果沒有用過這個類,可以按照下面的方式來調(diào)試看看一個表達式轉(zhuǎn)換為表達式樹是怎么樣的。
TestExpression(x=>x.Name == "test",x=>x.State==1);
public static void TestExpression(Expression<Func<Report, bool>> left,Expression<Func<Report, bool>> right)
{
//調(diào)試查看expression對象
var bodyLeft = left.Body;//這個就是x.Name == "test"
var bodyRight = right.Body;//這個就是x.State==1
}好,這里我們能獲取到表達式的Body,然后使用Expression類能很好的合并兩個表達式的body
var andAlso = Expression.AndAlso(bodyLeft ,bodyRight);//x.Name == "test" && x.State==1
這樣還不行,這兩個表達式是兩個不同的委托對象,他們的參數(shù)x也是兩個不同的對象,合并了又沒完全合并
這就需要用到ExpressionVisitor類來遞歸表達式樹,把兩個表達式的參數(shù)替換為同一個參數(shù)。
/// <summary>
/// 替換表達式參數(shù)
/// </summary>
public class ReplaceExpressionVisitor : ExpressionVisitor
{
private Expression _leftParameter;
public ReplaceExpressionVisitor(Expression leftParameter)
{
_leftParameter= leftParameter;
}
protected override Expression VisitParameter(ParameterExpression node)
{
return _leftParameter;
}
}
最終
TestExpression(x=>x.Name == "test",x=>x.State==1);
public static void TestExpression(Expression<Func<Report, bool>> left,Expression<Func<Report, bool>> right)
{
//調(diào)試查看expression對象
var bodyLeft = left.Body;//這個就是x.Name == "test"
var bodyRight = right.Body;//這個就是x.State==1
var leftParameter = left.Parameters[0];
//表達式遞歸訪問
var visitor =new ReplaceExpressionVisitor(leftParameter);
//替換參數(shù)
bodyRight = visitor.Visit(bodyRight);
//合并表達式
var expression = Expression.AndAlso(bodyLeft , bodyRight);
//構(gòu)建表達式
var whereExpression= Expression.Lambda<Func<Report, bool>>(expression , left.Parameters);
//編譯表達式
var whereFilter = whereExpression.Compile();
//使用
DbContext.Report.Where(whereFilter);
}正想給小老弟顯擺一下的時候,他又去寫其他騷代碼了
騷不過騷不過,完善一下列子,下面是完整的代碼
小嫩手不想動的小伙伴可以直接nuget上查找DynamicExpression.Core,直接使用
更多源碼看本人github
/// <summary>
/// 替換表達式參數(shù)
/// </summary>
public class ReplaceExpressionVisitor : ExpressionVisitor
{
private Dictionary<Expression, Expression> _parameters;
public ReplaceExpressionVisitor(Dictionary<Expression,Expression> parameters)
{
_parameters = parameters;
}
protected override Expression VisitParameter(ParameterExpression node)
{
if (_parameters.TryGetValue(node, out Expression _newValue))
{
return _newValue;
}
return base.Visit(node);
}
} /// <summary>
/// 表達式擴展
/// </summary>
public static class ExpressionExtension
{
/// <summary>
/// 使用AndAlso合并表達式
/// </summary>
/// <param name="exprs"></param>
/// <returns></returns>
public static Expression<T> AndAlso<T>(this IList<Expression<T>> exprs)
{
if (exprs.Count == 0) return null;
if (exprs.Count == 1) return exprs[0];
var leftExpr = exprs[0];
var left = leftExpr.Body;
for (int i = 1; i < exprs.Count; i++)
{
var expr = exprs[i];
var visitor = GetReplaceExpressionVisitor(expr.Parameters, leftExpr.Parameters);
var right = visitor.Visit(expr.Body);
left = Expression.AndAlso(left, right);
}
return Expression.Lambda<T>(left, leftExpr.Parameters);
}
/// <summary>
/// 使用AndAlso合并表達式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="left"></param>
/// <param name="right"></param>
/// <returns>left AndAlso right</returns>
public static Expression<T> AndAlso<T>(this Expression<T> left, Expression<T> right)
{
return AndAlso(new List<Expression<T>>() { left, right });
}
/// <summary>
/// 使用OrElse合并表達式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="exprs"></param>
/// <returns></returns>
public static Expression<T> OrElse<T>(this IList<Expression<T>> exprs)
{
if (exprs.Count == 0) return null;
if (exprs.Count == 1) return exprs[0];
var leftExpr = exprs[0];
var left = leftExpr.Body;
for (int i = 1; i < exprs.Count; i++)
{
var expr = exprs[i];
var visitor = GetReplaceExpressionVisitor(expr.Parameters, leftExpr.Parameters);
var right = visitor.Visit(expr.Body);
left = Expression.OrElse(left, right);
}
return Expression.Lambda<T>(left, leftExpr.Parameters);
}
/// <summary>
/// 使用OrElse合并表達式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="left"></param>
/// <param name="right"></param>
/// <returns>left OrElse right</returns>
public static Expression<T> OrElse<T>(this Expression<T> left, Expression<T> right)
{
return OrElse(new List<Expression<T>>() { left, right });
}
/// <summary>
/// 構(gòu)建visitor
/// </summary>
/// <param name="oldParameters"></param>
/// <param name="newParameters"></param>
/// <returns></returns>
private static ReplaceExpressionVisitor GetReplaceExpressionVisitor(ReadOnlyCollection<ParameterExpression> oldParameters, ReadOnlyCollection<ParameterExpression> newParameters)
{
Dictionary<Expression, Expression> dic = new Dictionary<Expression, Expression>();
for (int i = 0; i < oldParameters.Count; i++)
{
dic.Add(oldParameters[i],newParameters[i]);
}
return new ReplaceExpressionVisitor(dic);
}
}
使用
string connectString = "Data Source=.;Initial Catalog=RportTest;Integrated Security=True";
var optionsBuilder = new DbContextOptionsBuilder<TestContext>();
optionsBuilder.UseSqlServer(connectString);
using (TestContext ctx = new TestContext(optionsBuilder.Options))
{
Expression<Func<ReportData, bool>> epxr1 = report => report.ID == 2023;
Expression<Func<ReportData, bool>> epxr2 = report => report.Name == "test1";
var epxr3 = new List<Expression<Func<ReportData, bool>>>() { epxr1, epxr2 };
var andPredicate = epxr3.AndAlso();
var andQuery = ctx.ReportData.Where(andPredicate);
string andSql = andQuery.ToQueryString();
var andResult = andQuery.ToList();
var orPredicate = epxr3.OrElse();
var orQuery = ctx.ReportData.Where(orPredicate);
string orSql = orQuery.ToQueryString();
var orResult = orQuery.ToList();
}到此這篇關(guān)于.net lambda表達式合并的文章就介紹到這了,更多相關(guān).net lambda表達式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
ASP.NET MVC5使用MiniProfiler監(jiān)控MVC性能
這篇文章主要為大家詳細(xì)介紹了ASP.NET MVC5使用MiniProfiler監(jiān)控MVC性能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07
Entity Framework Core實現(xiàn)軟刪除與查詢過濾器
這篇文章介紹了Entity Framework Core實現(xiàn)軟刪除與查詢過濾器的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-02-02
Asp.net Socket客戶端(遠(yuǎn)程發(fā)送和接收數(shù)據(jù))
通過Socket遠(yuǎn)程發(fā)送與接收數(shù)據(jù)的代碼類2008-11-11
.NET Core跨平臺串口通訊使用SerialPortStream基礎(chǔ)類庫問題解決
這篇文章介紹了.NET Core跨平臺串口通訊使用SerialPortStream基礎(chǔ)類庫問題解決,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-01-01
DropDownList根據(jù)下拉項的Text文本序號排序
在某些時候表中沒有可以排序的字段同時呢也不想修改表結(jié)構(gòu),但它的項文本有序號這時就可以用這方法排序,感興趣的你可以參考下,或許本文知識點對你有所幫助2013-03-03
ASP.NET設(shè)計網(wǎng)絡(luò)硬盤之查看文件夾實現(xiàn)代碼
下面要介紹的實例包括網(wǎng)上硬盤的許多功能,將一步步為大家進行介紹。首先創(chuàng)建工程實例,然后進行主界面的設(shè)計,最后對各個功能的實現(xiàn)分別進行介紹2012-10-10

