如何使用<dynamic>动态构建的 lambda 表达式搜索动态对象 IEnumerable?



以前,我对我之前的问题有很大帮助,谢谢 vyrp,如何使用动态构建的 lambda 表达式创建和填充动态对象

我现在正在寻找搜索动态对象,和以前一样,我不知道对象属性,因此在运行时之前我正在搜索什么。

下面是生成动态对象的代码:

        // Get list of optional fields
        var optFieldList = await _tbList_FieldRepository.GetAsync(lf => lf.ListID == listId && lf.DisplayInList == true);
        // order list of optional fields
        optFieldList.OrderBy(lf => lf.DisplayOrder);
        // Get base Data excluding Inactive if applicable
        IEnumerable<tbList_Data> primaryData = await _tbList_DataRepository.GetAsync(ld => ld.ListID == listId && (ld.IsActive == includeInactive ? ld.IsActive : true));
        // Build IEnumerable<dynamic> from base results plus any optional fields to be displayed in table   
        var results = primaryData.Select(pd => {
            dynamic result = new System.Dynamic.ExpandoObject();
            result.Id = pd.ID;
            result.PrimaryData = pd.PrimaryData;
            result.DisplayOrder = pd.DisplayOrder;
            result.IsActive = pd.IsActive;
            foreach (var optField in optFieldList)
            {
                switch (optField.FieldType.ToLower()) {
                    case "text":
                        ((IDictionary<string, object>)result).Add(optField.FieldName, pd.tbList_DataText.Where(ld => ld.DataRowID == pd.ID && ld.ListColumnID == optField.ID).Select(ld => ld.DataField).DefaultIfEmpty("").First());
                        break;
                }
            }
            return result;
        });

为了进行测试,我有 2 个动态字段,"电话号码"和"燃料类型">

我可以搜索已知字段,即主要数据,没问题,如下所示。

        results = results.Where(r => r.PrimaryData.Contains(searchString));

如果我在设计时知道字段电话号码,以下内容将起作用

        results = results.Where(r => r.PhoneNumber.Contains(searchString));

但我想做的是这样的:

results = results.Where(r => r.PrimaryData.Contains(searchString) 
    || foreach(var optField in optFieldList)
    {
        r.optField.FieldName.Contains(searchString)
    })

最终以

results = results.Where(r => 
    r.PrimaryData.Contains(searchString) 
    || r.PhoneNumber.Contains(searchString) ||    
    r.FuelType.Contains(searchString));

但显然该代码不起作用。 我尝试了很多不同的尝试,但没有一次成功,所以我正在寻找建议。 谢谢

由于您知道查询的 dynamic 元素实际上是 ExpandoObject ,因此IDictionary<string, object>>,您可以安全地将其强制转换为字典接口并使用它来按名称访问属性值,而 Enumerable.任何方法都可用于模拟动态||条件:

results = results.Where(r => r.PrimaryData.Contains(searchString)
    || optFieldList.Any(f =>
    {
        object value;
        return ((IDictionary<string, object>)r).TryGetValue(f.FieldName, out value)
            && value is string && ((string)value).Contains(searchString);
    }));

最新更新