使用LINQ按字符串名称选择属性

本文关键字:选择 属性 字符串 LINQ 使用 | 更新日期: 2023-09-27 17:53:40

我有一段非常简单的代码:

var list = new List<MyType>();
list = MyItems.Where(x => x.Name.ToLower().Contains(e.Text.ToLower())).ToList();
for (int i = itemOffset; i < endOffset; i++)
{
    combo.Items.Add(new ComboItem(list[i].Name, list[i].Id.ToString()));
}

我想做的是使它成为一个泛型函数,它接受任何类型的列表,并接受两个字符串作为属性。(在这种情况下,我们正在查看的两个属性是Name和Id。

private void MakeCombo<T>(Combo combo, IEnumerable<T> lst, string Field1, string Field2)
{
    var list = list = lst.Where(x => x.Field1.ToLower().Contains(searchText.ToLower())).ToList(); //How make this to work???
    for (int i = itemOffset; i < endOffset; i++)
    {
        combo.Items.Add(new ComboItem(list[i].Field1, list[i].Field2.ToString())); //How make this to work???
    }
}

现在,我不明白如何通过字符串名称(Field1和Field2)访问通用列表的属性。

使用LINQ按字符串名称选择属性

如果您无法创建接口并确保序列中的所有项都实现该接口(如另一个答案所示),那么解决此问题的一种方法是使用委托。通过传入两个函数,每个函数都以T作为输入,并返回stringobject作为输出,您可以模拟该功能:

private void MakeCombo<T>(Combo combo, IEnumerable<T> sequence
    , Func<T, string> field1Selector, Func<T, object> field2Selector)
{
    var foundItems = sequence.Where(item => field1Selector(item).ToLower()
        .Contains(searchText.ToLower()));
    foreach(var item in foundItems)
    {
        combo.Items.Add(new ComboItem(field1Selector(item), field2Selector(item).ToString()));
    }
}

从调用端来看,它可能看起来像这样(以您的第一个代码片段为例:

MakeCombo(combo, list, item => item.Name, item => item.Id);

你必须使用反射:

typeof(x).GetField("Field1", BindingFlags.Public).GetValue()

您可以对两个对象都这样做,并比较值

如果您可以定义一个接口,描述所有可用的字段,并使每个类型适合于此方法实现该接口,它将看起来像这样(并且将比通过名称访问字段更好):

public interface IFields
{
    string Prop1 { get; set; }
    string Prop2 { get; set; }
}
public static void DoSmth<T>(List<T> items) where T : IFields
{
    foreach (T t in items)
    {
        Console.WriteLine(t.Prop1);
        Console.WriteLine(t.Prop2);
    }
}
private void MakeCombo<T>(Combo combo, IEnumerable<T> list)
{
    Type type = typeof(T);
    List<T> comboList = list.Where(x => ((String)type.GetField("Field1", (BindingFlags.Public | BindingFlags.Instance)).GetValue(x)).ToLower().Contains(m_SearchText.ToLower()));
    for (Int32 i = m_ItemOffset; i < m_OffsetEnd; ++i)
        combo.Items.Add(new ComboItem((String)type.GetField("Field1", (BindingFlags.Public | BindingFlags.Instance)).GetValue(list[i]), (String)type.GetField("Field2", (BindingFlags.Public | BindingFlags.Instance)).GetValue(list[i])));
}