电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

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

瀏覽:157日期: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
主站蜘蛛池模板: 冷却塔改造厂家_不锈钢冷却塔_玻璃钢冷却塔改造维修-广东特菱节能空调设备有限公司 | 水质监测站_水质在线分析仪_水质自动监测系统_多参数水质在线监测仪_水质传感器-山东万象环境科技有限公司 | 滚珠丝杆升降机_螺旋升降机_丝杠升降机-德迈传动 | 远程会诊系统-手术示教系统【林之硕】医院远程医疗平台 | 旋转/数显粘度计-运动粘度测定仪-上海平轩科学仪器 | 爱德华真空泵油/罗茨泵维修,爱发科-比其尔产品供应东莞/杭州/上海等全国各地 | 南京租车,南京汽车租赁,南京包车,南京会议租车-南京七熹租车 | 水质监测站_水质在线分析仪_水质自动监测系统_多参数水质在线监测仪_水质传感器-山东万象环境科技有限公司 | 水冷散热器_水冷电子散热器_大功率散热器_水冷板散热器厂家-河源市恒光辉散热器有限公司 | 扒渣机,铁水扒渣机,钢水扒渣机,铁水捞渣机,钢水捞渣机-烟台盛利达工程技术有限公司 | 锂电混合机-新能源混合机-正极材料混料机-高镍,三元材料混料机-负极,包覆混合机-贝尔专业混合混料搅拌机械系统设备厂家 | 深圳彩钢板_彩钢瓦_岩棉板_夹芯板_防火复合彩钢板_长鑫 | 起好名字_取个好名字_好名网免费取好名在线打分 | 解放卡车|出口|济南重汽|报价大全|山东三维商贸有限公司 | 德国UST优斯特氢气检漏仪-德国舒赐乙烷检测仪-北京泽钏 | 鼓风干燥箱_真空烘箱_高温干燥箱_恒温培养箱-上海笃特科学仪器 | 高压管道冲洗清洗机_液压剪叉式升降机平台厂家-林君机电 | 大通天成企业资质代办_承装修试电力设施许可证_增值电信业务经营许可证_无人机运营合格证_广播电视节目制作许可证 | 右手官网|右手工业设计|外观设计公司|工业设计公司|产品创新设计|医疗产品结构设计|EMC产品结构设计 | 【电子厂招聘_普工招工网_工厂招聘信息平台】-工立方打工网 | 颚式破碎机,圆锥破碎机,制砂机-新乡市德诚机电制造有限公司 | 活性氧化铝球|氧化铝干燥剂|分子筛干燥剂|氢氧化铝粉-淄博同心材料有限公司 | 蜂窝块状沸石分子筛-吸附脱硫分子筛-萍乡市捷龙环保科技有限公司 | 锂电混合机-新能源混合机-正极材料混料机-高镍,三元材料混料机-负极,包覆混合机-贝尔专业混合混料搅拌机械系统设备厂家 | 我爱古诗词_古诗词名句赏析学习平台 | 体检车_移动CT车_CT检查车_CT车_深圳市艾克瑞电气有限公司移动CT体检车厂家-深圳市艾克瑞电气有限公司 | 亚克力制品定制,上海嘉定有机玻璃加工制作生产厂家—官网 | 胶水,胶粘剂,AB胶,环氧胶,UV胶水,高温胶,快干胶,密封胶,结构胶,电子胶,厌氧胶,高温胶水,电子胶水-东莞聚力-聚厉胶粘 | 代做标书-代写标书-专业标书文件编辑-「深圳卓越创兴公司」 | 塑料检查井_双扣聚氯乙烯增强管_双壁波纹管-河南中盈塑料制品有限公司 | 权威废金属|废塑料|废纸|废铜|废钢价格|再生资源回收行情报价中心-中废网 | 南京种植牙医院【官方挂号】_南京治疗种植牙医院那个好_南京看种植牙哪里好_南京茀莱堡口腔医院 尼龙PA610树脂,尼龙PA612树脂,尼龙PA1010树脂,透明尼龙-谷骐科技【官网】 | 拼装地板,悬浮地板厂家,悬浮式拼装运动地板-石家庄博超地板科技有限公司 | 电镀整流器_微弧氧化电源_高频电解电源_微弧氧化设备厂家_深圳开瑞节能 | 贴片电容-贴片电阻-二三极管-国巨|三星|风华贴片电容代理商-深圳伟哲电子 | 广西正涛环保工程有限公司【官网】| 不锈钢轴流风机,不锈钢电机-许昌光维防爆电机有限公司(原许昌光维特种电机技术有限公司) | 鑫达滑石-辽宁鑫达滑石集团 | 广东教师资格网-广东教师资格证考试网| 一体化污水处理设备,一体化污水设备厂家-宜兴市福源水处理设备有限公司 | 照相馆预约系统,微信公众号摄影门店系统,影楼管理软件-盟百网络 |