如何在C#中将多个函数作为约束参数传递?

时间:2013-01-29 13:34:37

标签: c# linq .net-4.0 methods delegates

对于我正在努力学习的内容,这可能是一个措辞严厉的问题,所以我将解释我想要做的事情。我使用以下表达式创建了以下列表:

var showIndexes = from item in lItems
                               where WildcardString.IsMatch(item.RecordId.ToString(), filter) ||
                                     WildcardString.IsMatch(item.Tokens.ToString(), filter) ||
                                     WildcardString.IsMatch(string.Format("{0}:{1}", item.OwnerID.SystemName, item.OwnerID.Port.ToString()), filter) ||
                                     WildcardString.IsMatch(item.ChildID.UserName, filter) ||
                                     WildcardString.IsMatch(item.TypeOfLicense.ToString(), filter) ||
                                     WildcardString.IsMatch(item.ExpiryTime.DateTime.ToLocalTime().ToString(), filter)
                               select item.RecordId.ToString();

我想将此表达式放入函数中,并将“where”中使用的约束作为单个参数传递。我该如何实现这一目标?我在使用文件属性时(当你选择不同的属性枚举时)看到了使用按位OR的类似功能,所以我认为应该可以编写我自己的函数,它将一组代理作为“where”规则。

如果相关,WildcardString.IsMatch(string,string)是一个布尔函数。此外,我有三个不同的函数执行完全相同的事情,但在不同的列表上,这就是为什么我想学习如何将此减少到单个函数,并只是传递列表和选择规则。

3 个答案:

答案 0 :(得分:3)

简单地定义一个谓词:一个接受item并返回布尔值的函数。

假设item类型为Recordfilter在范围内:

Func<Record, bool> predicate = r =>
    WildCardString.IsMatch(r.RecordId.ToString(), filter) ||
    WildCardString.IsMatch(...);

然后,您可以将predicate视为第一类值(存储它,将其作为参数传递等)并使用

过滤任何IEnumerable<Record>
var filtered = enumerable.Where(predicate);

答案 1 :(得分:2)

只需将where部分重构为函数:

public bool MyIsMatch(MyType item, String filter)
{
    return
    WildcardString.IsMatch(item.RecordId.ToString(), filter) ||
    WildcardString.IsMatch(item.Tokens.ToString(), filter) ||
    WildcardString.IsMatch(string.Format("{0}:{1}", item.OwnerID.SystemName, item.OwnerID.Port.ToString()), filter) ||
    WildcardString.IsMatch(item.ChildID.UserName, filter) ||
    WildcardString.IsMatch(item.TypeOfLicense.ToString(), filter) ||
    WildcardString.IsMatch(item.ExpiryTime.DateTime.ToLocalTime().ToString(), filter);
}

然后叫它:

var showIndexes = from item in lItems where MyIsMatch(item, filter) select item.ToString();

答案 2 :(得分:1)

您可以创建一个带有有效值列表的Is.Match版本

public static bool IsMatch(IEnumerable<object> values, filter) {
    return values.All(v => IsMatch(v.ToString(),filter))
}

在你的情况下你会像这样使用它

var values = new object[]{
                    item.RecordId,
                    item.Tokens,
                    string.Format("{0}:{1}", item.OwnerID.SystemName, item.OwnerID.Port),
                    item.ChildID.UserName
                    item.TypeOfLicense
                    item.ExpiryTime.DateTime};

var showIndexes = from item in lItems
                  where WildcardString.IsMatch(values,filter)
                           select item.RecordId.ToString();

我选择使用IEnumerable<object>代替IEnumerable<string>从每个项目中删除.ToString(),但这会带来很大的成本,因为您将装箱并取消装箱任何ValueTyped值。如果它位于已确定的性能关键路径上,则应使用IEnumerable<string>代替