3

私は非常に単純なコードを持っています:

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

私がやりたいのは、これを任意のタイプのリストを受け入れ、プロパティとして2つの文字列を受け入れるジェネリック関数にすることです。(この場合、私たちが見ている2つのプロパティは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)でジェネリックリストのプロパティにアクセスする方法がわかりません。

4

4 に答える 4

4

この問題を解決する1つの方法は、インターフェイスを作成し、シーケンス内のすべてのアイテムがそのインターフェイスを実装することを保証する機能がない場合(この他の回答に示されているように)、デリゲートを使用することです。それぞれが入力として、またはを出力としてT返す2つの関数を渡すことにより、その機能を模倣できます。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);
于 2013-01-16T05:30:56.120 に答える
3

リフレクションを使用する必要があります。

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

両方のオブジェクトに対してそれを行い、値を比較することができます。

于 2013-01-16T00:51:28.737 に答える
1

使用可能なすべてのフィールドを記述してインターフェースを定義し、このメソッドに適した各タイプにそのインターフェースを実装させることができれば、次のようになります(名前でフィールドにアクセスするよりもIMOの方が優れています)。

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);
    }
}
于 2013-01-16T00:51:45.283 に答える
1
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])));
}
于 2013-01-16T01:04:22.347 に答える