对于我正在努力学习的内容,这可能是一个措辞严厉的问题,所以我将解释我想要做的事情。我使用以下表达式创建了以下列表:
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)是一个布尔函数。此外,我有三个不同的函数执行完全相同的事情,但在不同的列表上,这就是为什么我想学习如何将此减少到单个函数,并只是传递列表和选择规则。
答案 0 :(得分:3)
简单地定义一个谓词:一个接受item
并返回布尔值的函数。
假设item
类型为Record
且filter
在范围内:
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>
代替