优雅的方式从对象列表到具有两个属性的字典

时间:2009-06-01 11:59:39

标签: c# arrays visual-studio-2005 dictionary

我似乎一遍又一遍地编写这段代码,想看看是否有更好的方法来做到这一点。

我从一个Foo对象列表开始

Foo[] foos = GenerateFoos();

我想创建一个字典,其中键和值都是Foo

的属性

例如:

Dictionary<string, string> fooDict = new Dictionary<string, string>():
foreach (Foo foo in foos)
{
    fooDict[foo.Name] = foo.StreetAddress;
}

无论如何都要编写这个代码,因为它看起来像一个基本模板,其中有一个对象数组,一个键属性,一个值属性和一个字典。

有什么建议吗?

我正在使用VS 2005(C#,2.0)

4 个答案:

答案 0 :(得分:70)

使用LINQ:

var fooDict = foos.ToDictionary(x=>x.Name,x=>x.StreetAddress);

(是的,fooDictDictionary<string, string>


编辑以显示VS2005中的痛苦:

Dictionary<string, string> fooDict =
    Program.ToDictionary<Foo, string, string>(foos,
        delegate(Foo foo) { return foo.Name; },
        delegate(Foo foo) { return foo.StreetAddress; });

你在哪里(Program):

public static Dictionary<TKey, TValue> ToDictionary<TSource, TKey, TValue>(
    IEnumerable<TSource> items,
    Converter<TSource, TKey> keySelector,
    Converter<TSource, TValue> valueSelector)
{
    Dictionary<TKey, TValue> result = new Dictionary<TKey, TValue>();
    foreach (TSource item in items)
    {
        result.Add(keySelector(item), valueSelector(item));
    }
    return result;
}

答案 1 :(得分:7)

如果您使用的是框架3.5,则可以使用ToDictionary扩展程序:

Dictionary<string, string> fooDict = foos.ToDictionary(f => f.Name, f => f.StreetAddress);

对于框架2.0,代码非常简单。

您可以通过在创建字典时指定字典的容量来稍微提高性能,以便在填写字典时不必进行任何重新分配:

Dictionary<string, string> fooDict = new Dictionary<string, string>(foos.Count):

答案 2 :(得分:2)

没有LINQ,不,没有内置帮助器。你可以写一个:

// I forget if you need this delegate definition -- this may be already defined in .NET 2.0
public delegate R Func<T,R>(T obj);
public static Dictionary<K,V> BuildDictionary<T,K,V>(IEnumerable<T> objs, Func<T,K> kf, Func<T,V> vf)
{
    Dictionary<K,V> d = new Dictionary<K,V>();
    foreach (T obj in objs)
    {
        d[kf(obj)] = vf(obj);
    }
    return d;
}

Dictionary<string, string> fooDict = BuildDictionary(foos, new Func<Foo,string>(delegate(Foo foo) { return foo.Name; }), new Func<Foo,string>(delegate(Foo foo) { return foo.StreetAddress; }));

它看起来不像基于LINQ的答案那么优雅,是吗......

答案 3 :(得分:1)

这是一个与.net 2.0兼容的解决方案,它使用System.Web.UI.Databinder对属性名称进行反射 - 您将失去编译时类型检查。

        public static Dictionary<string, string> ToDictionary<T>(List<T> list, string keyName, string valueName)
    {
        Dictionary<string, string> outputDictionary = new Dictionary<string, string>();
        foreach (T item in list)
        {
            string key = Eval<T, string>(item, keyName);
            string value = Eval<T, string>(item, valueName);
            output[key] = value;
        }

        return outputDictionary;
    }

    public static TOut Eval<TIn, TOut>(TIn source, string propertyName)
    {
        object o = DataBinder.GetPropertyValue(source, propertyName);
        if (o is TOut)
            return (TOut)o;

        return default(TOut);
    }

你会打电话如下:

Dictionary<string, string> fooDict = ToDictionary(foos, "Name", "StreetAddress");