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

C# 使用LINQ按字符串名称选择属性,c#,.net,linq,list,C#,.net,Linq,List,我有一段非常简单的代码: 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.ToStrin

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

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()));
}
var list=newlist();
list=MyItems.Where(x=>x.Name.ToLower().Contains(e.Text.ToLower()).ToList();
对于(int i=itemOffset;i
我想做的是使它成为一个通用函数,它接受任何类型的列表,并接受两个字符串作为属性。(在本例中,我们正在查看的两个属性是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???
    }
}
private void MakeCombo(组合组合、IEnumerable lst、字符串字段1、字符串字段2)
{
var list=list=lst.Where(x=>x.Field1.ToLower().Contains(searchText.ToLower()).ToList();//如何使其工作???
对于(int i=itemOffset;i

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

您必须使用反射:

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

您可以对这两个对象执行此操作并比较值。

如果您可以定义一个接口,描述所有可用的字段,并使适合此方法的每种类型实现该接口,它将如下所示(并且比按名称访问字段更好):

公共接口IFields
{
字符串Prop1{get;set;}
字符串Prop2{get;set;}
}
公共静态无效数据(列表项),其中T:i字段
{
foreach(以项目为单位)
{
控制台写入线(t.Prop1);
控制台写入线(t.Prop2);
}
}
私有void MakeCombo(组合组合,IEnumerable列表)
{
类型=类型(T);
列表组合列表=列表。其中(x=>((字符串)type.GetField(“Field1”),(BindingFlags.Public | BindingFlags.Instance)).GetValue(x)).ToLower()包含(m_SearchText.ToLower());
对于(Int32 i=m_ItemOffset;i
解决此问题的一种方法,如果您无法创建接口并确保序列中的所有项都实现该接口(如所示),是使用委托。通过传入两个函数,每个函数以
T
作为输入,并返回
字符串
对象
作为输出,可以模拟该功能:

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()));
    }
}

什么不起作用?是否有错误?是否有有用的转储?您正在为列表中的每个项目计算searchText.ToLower()。您应该存储到一个局部变量,以便只需计算一次。通常,您应该尽可能避免这样做(在这里也是可能的)。反射应该始终是最后的手段,就像任何其他将字符串视为代码的方法一样。反射非常脆弱,会妨碍学习更合适的编码方法(您可以开始将其用作拐杖),会造成安全漏洞,并且随着时间的推移,更难阅读/理解/维护。这看起来很棒。不幸的是,我无法描述接口,因为我对T没有真正的控制权。没有接口有什么方法可以做到吗?@user194076请使用Chris的答案then@user194076是的,有。请让两名代表进来,例如de>Func
。这将是文本和值的选择器。调用该方法时,可以使用lambda,例如
item=>item.Name
。与反射模型不同,这具有100%编译时安全性的优点,因为您不必担心是否存在具有给定名称的字段、字段的可见性等,您可以n只需使用函数。@Servy您可能想使用与此问题中类似的代码:?@KonstantinVasilcov不,这里不需要表达式。这比那简单。我想我会把它写下来作为一个答案。Get value至少需要一个参数。它应该是什么?哇,您应该传递ty的实例pe您想要的值为。在本例中为x。如果您想沿着这条路线走下去,您应该认真阅读有关反射的内容。它抱怨说:在同一个类型中,找不到类型或命名空间名称“x”(您是否缺少using指令或程序集引用?),而类型为(列表[i])好的,它现在可以编译了。我遇到的问题是:type.GetField(“Name”,BindingFlags.NonPublic | BindingFlags.Instance)-它返回null。而这个:type.GetFields(BindingFlags.NonPublic | BindingFlags.Instance)返回={System.Reflection.FieldInfo[2]}[0]:{Int32 k_ubackingfield}[1]:{System.String k_ubackingfield}你知道为什么会发生这种情况吗?你能给我看看那些字段声明的代码片段吗?比如:`[Serializable()]公共类MyClass{public MyClass(){….}[Serializable()]公共类MyNestedClass{public int Id{get;set;}公共字符串名称{get;set;}公共MyNestedClass(){}公共MyNestedClass(int val0,string val1){Id=val0;Name=val1;}`非常感谢你写下这个答案。这太棒了。
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])));
}
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);