如何创建一个c#方法,它将返回与具有不同元素类型的参数相同的集合类型?

时间:2016-07-12 15:54:21

标签: c# generics

基本上,与this question完全相同,但在c#中。

我希望有这样的东西:

public IList<double> DoSomething(IList<string> strings) {
    //something
}

但是我宁愿让它输入,以便提前知道返回值,这样我就不需要在函数之外进行强制转换。我该怎么做?

是否存在通用方法,

public ListType<double> DoSomething<ListType>(ListType<string> strings) where ListType : ???{

设置它?

2 个答案:

答案 0 :(得分:2)

C#不允许你想要的,但最接近的是:

public TList2 DoSomething<TList1, TList2>(TList1 strings) 
    where TList1 : IEnumerable<string>
    where TList2 : ICollection<decimal>, new()
{
    var result = new TList2();
    foreach (var s in strings)
    {
        result.Add(decimal.Parse(s));
    }
    return result;
}

和一个例子:

var input = new HashSet<string>(new[] { "1", "2", "3"});
List<decimal> output = DoSomething<HashSet<string>, List<decimal>>(input);

答案 1 :(得分:0)

您可以使用Linq执行此操作。

例如,如果你只想解析加倍,你会这样做:

List<double> result = strings.Select(double.Parse).ToList()

你可以发送任何其他方法而不是double.Parse:

List<double> result = strings.Select(DoSomethingWithOneItem).ToList()

double DoSomethingWithOneItem(string item) {
  //your conversion logic
}

不幸的是,例如IList<TInput>IList<TOutput>之间没有任何关系可以用来帮助这里,所以你需要同时指定输入和输出列表类型,这有点麻烦一般形式:

IList<string> strings  = new List<string> {"1.1", "2.2", "3.3"};
IList<decimal> result = strings.ConvertToSameListType((Func<string, decimal>)decimal.Parse, () => new List<decimal>());

public static class EnumerableExtensioncGeneralVersion
{
    public static TOutputList ConvertToSameListType<TInputList, TOutputList, TInput, TOutput>(this TInputList source, Func<TInput, TOutput> itemConversion, Func<TOutputList> outputListConstructor)
        where TInputList : IEnumerable<TInput>
        where TOutputList : ICollection<TOutput>
    {
        TOutputList result = outputListConstructor();
        foreach (TOutput convertedItem in source.Select(itemConversion))
        {
            result.Add(convertedItem);
        }
        return result;
    }
}

如果您只想为要支持的每种喜爱的收藏类型指定一种扩展方法,那么您可以更好地使用转化:

//Seting up inputs
IList<string> strings  = new List<string> {"1.1", "2.2", "3.3"};
IEnumerable<string> enumerableStrings = strings.Select(x => x);
ObservableCollection<string> observableStrings = new ObservableCollection<string>(strings);

//Converting
IList<decimal> resultList = strings.Convert(decimal.Parse);
IEnumerable<decimal> resultEnumerable = enumerableStrings.Convert(decimal.Parse);
ObservableCollection<decimal> observableResult = observableStrings.Convert(decimal.Parse);

public static class EnumerableExtensions
{
    public static IList<TOutput> Convert<TInput, TOutput>(this IList<TInput> source, Func<TInput, TOutput> itemConversion)
    {
        return source.Select(itemConversion).ToList();
    }
    public static IEnumerable<TOutput> Convert<TInput, TOutput>(this IEnumerable<TInput> source, Func<TInput, TOutput> itemConversion)
    {
        return source.Select(itemConversion);
    }
    public static ObservableCollection<TOutput> Convert<TInput, TOutput>(this ObservableCollection<TInput> source, Func<TInput, TOutput> itemConversion)
    {
        return new ObservableCollection<TOutput>(source.Select(itemConversion));
    }
}