日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

.NET中lambda表達式合并問題及解決方法

瀏覽:371日期:2022-06-08 15:25:19
目錄
  • 解決方案:
  • 完美解決

事情的起因是公司一個小伙子問了我個問題 “海哥,來幫我看下這段代碼怎么不行”

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是直接執行表達式,他就是個委托方法,里面嵌套多少層委托和方法都是能直接執行的
IQueryable并不會執行表達式和方法,是把表達式轉換為對應的Sql語句來執行,解析到nameFilter的時候他就懵逼了,這是啥玩意兒啊,sql里面沒有這種東西啊,他就轉換不了了。

小伙子知道后明細很失望,那不能啊,也不是我想顯擺我的技術,就是想讓小伙子能繼續他的騷操作,給他來點海克斯科技與狠活。

解決方案:

//表達式
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;

//調用
DbContext.Report.Where(whereFilter);

完美解決

那怎么合并,當然得自己構造一個新的表達式,構造表達式需要用到Expression類,如果沒有用過這個類,可以按照下面的方式來調試看看一個表達式轉換為表達式樹是怎么樣的。

TestExpression(x=>x.Name == "test",x=>x.State==1);

public static void TestExpression(Expression<Func<Report, bool>> left,Expression<Func<Report, bool>> right)
{ 
    //調試查看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

這樣還不行,這兩個表達式是兩個不同的委托對象,他們的參數x也是兩個不同的對象,合并了又沒完全合并

這就需要用到ExpressionVisitor類來遞歸表達式樹,把兩個表達式的參數替換為同一個參數。

    /// <summary>
    /// 替換表達式參數
    /// </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)
{ 
    //調試查看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);
    //替換參數
    bodyRight = visitor.Visit(bodyRight);
    //合并表達式
    var expression = Expression.AndAlso(bodyLeft , bodyRight);
    //構建表達式
    var whereExpression= Expression.Lambda<Func<Report, bool>>(expression , left.Parameters);
    //編譯表達式
    var whereFilter = whereExpression.Compile();
    //使用
    DbContext.Report.Where(whereFilter);
}

正想給小老弟顯擺一下的時候,他又去寫其他騷代碼了

騷不過騷不過,完善一下列子,下面是完整的代碼

小嫩手不想動的小伙伴可以直接nuget上查找DynamicExpression.Core,直接使用

更多源碼看本人github

    /// <summary>
    /// 替換表達式參數
    /// </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>
/// 構建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();
}

到此這篇關于.net lambda表達式合并的文章就介紹到這了,更多相關.net lambda表達式內容請搜索以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持!

標簽: ASP.NET
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产一区国产二区国产三区| 一区久久精品| 欧美精品自拍| 久久久久久久久丰满| 精品国产aⅴ| 久久影院一区二区三区| 欧美日一区二区三区在线观看国产免 | 亚洲丝袜美腿一区| 99精品美女| 激情婷婷欧美| 激情亚洲影院在线观看| 在线亚洲人成| 欧美精选视频一区二区| 国产精品久久久久久久免费软件| 久久国产免费看| 国产精品成人3p一区二区三区| 国产免费播放一区二区| 国产乱子精品一区二区在线观看| 国产精品一卡| 国产在视频一区二区三区吞精| 国产成人精品亚洲线观看| 欧美激情国产在线| 激情国产在线| 国产综合精品| 美女精品一区| 日本不卡视频在线| 国产精品**亚洲精品| 9999国产精品| 国产精品久久观看| 欧美精品资源| 国产农村妇女精品一区二区 | 婷婷综合六月| 亚洲一区二区网站| 亚洲久久在线| 欧美激情三区| 色老板在线视频一区二区| 伊人精品视频| 日韩1区2区3区| 粉嫩av一区二区三区四区五区 | 韩日一区二区| 蜜桃精品在线| 亚洲婷婷丁香| 精品免费在线| 日韩午夜黄色| 国产精品自在| 成人羞羞视频播放网站| 一本综合精品| 日产午夜精品一线二线三线| 国产尤物精品| 日本不卡一二三区黄网| 国产不卡一区| 亚洲精品一区二区在线看| 蜜桃久久久久久久| 国产视频一区二| caoporn视频在线| 久久性天堂网| 久久99国产精品视频| 99成人在线| 麻豆精品在线播放| 亚洲天堂黄色| 91精品视频一区二区| 美女视频网站久久| 成人午夜国产| 国产乱子精品一区二区在线观看| 日本精品影院| 国产精品久久久久久久久久久久久久久 | 99精品视频在线观看免费播放| 蜜桃免费网站一区二区三区| 精品99在线| 综合一区二区三区| 香蕉视频亚洲一级| 日本成人手机在线| 国产一区亚洲| 精品国产91| 日产欧产美韩系列久久99| 日韩在线第七页| 婷婷综合一区| 久久狠狠婷婷| 青青草91久久久久久久久| 成人在线视频免费看| 中文字幕成人| 91视频一区| 蜜臀a∨国产成人精品| sm久久捆绑调教精品一区| 少妇高潮一区二区三区99| 天堂资源在线亚洲| 麻豆一区二区三| 日韩中文字幕一区二区高清99| 亚洲v在线看| 国产精品九九| 综合激情视频| 红桃视频国产精品| 高清不卡亚洲| 国产精品多人| 日韩精品欧美大片| 亚洲黄页一区| 午夜精品成人av| 精品成av人一区二区三区| 69堂免费精品视频在线播放| 人人精品人人爱| 女人av一区| 日本一二区不卡| 精品午夜久久| 麻豆精品蜜桃视频网站| 日本色综合中文字幕| 蜜桃视频在线观看一区| 午夜av一区| 免费视频国产一区| 精品免费av在线| 日韩国产欧美| 日韩不卡手机在线v区| 天堂成人免费av电影一区| 久久五月天小说| 成人小电影网站| 精品五月天堂| 欧美激情麻豆| 国产精品一区2区3区| 青青草国产成人99久久| 日本在线观看不卡视频| 综合在线一区| 亚洲乱亚洲高清| 日韩三区四区| 日韩午夜视频在线| 日韩欧美三区| 91成人精品在线| 一本综合精品| 天堂精品久久久久| 中文字幕成人| 日本欧美一区| 国产精品国产三级在线观看| 国产精品a级| 精品国产亚洲日本| 成人影视亚洲图片在线| 岛国av免费在线观看| 国产一区二区三区不卡av| 精品美女在线视频| 国产在线一区不卡| 精品国产免费人成网站| 久久久久久一区二区| 日韩成人亚洲| аⅴ资源天堂资源库在线| 国产精品蜜芽在线观看| 青青久久av| 亚洲一卡久久| 日韩中文字幕视频网| 欧美日韩亚洲一区在线观看| 国产欧美高清视频在线| 精品视频在线观看网站| 福利在线一区| 亚洲小说欧美另类婷婷| 国产亚洲毛片| 色8久久久久| 国产精品99精品一区二区三区∴| 精品国产乱码久久久久久樱花 | 日本精品久久| 国产欧美日韩在线观看视频| 久久精品一区二区国产| 国产伦久视频在线观看| 欧美日韩在线观看视频小说| 亚洲一区区二区| 蜜臀av在线播放一区二区三区| 日本一区二区三区视频在线看| 国产精品观看| 久久人人97超碰国产公开结果| 亚洲欧美日本国产专区一区| 日韩精品视频中文字幕| 麻豆精品久久| 在线日韩av| 尹人成人综合网| 亚洲91网站| 国产一区调教| 一区二区亚洲精品| 国产亚洲久久| 欧美成a人国产精品高清乱码在线观看片在线观看久 | 在线天堂资源www在线污| 亚洲精品.com| 人人爽香蕉精品| 久久精品一区二区国产| 1000部精品久久久久久久久| 亚洲免费毛片| 高潮久久久久久久久久久久久久| 欧美日韩视频| 国产欧美一区二区三区精品观看| 日本在线高清| 免费观看在线综合色| 九九九精品视频| 蜜桃视频欧美| 日本不卡一二三区黄网| 蜜桃精品视频| 黄色亚洲在线| 国产精品视频一区二区三区综合| 日韩精品影视| 日韩av一二三| 999精品色在线播放| 欧美片第1页综合| 欧美+亚洲+精品+三区| 国产精品久久亚洲不卡| 午夜视频精品| 激情中国色综合| 亚欧成人精品|