为什么我的通用扩展方法说不能从T转换为.....或....转换为T.

时间:2012-04-21 06:58:24

标签: c# .net extension-methods

继续我的问题

https://stackoverflow.com/questions/10241345/any-percentage-function-in-net

https://stackoverflow.com/questions/9942202/percentage-of-each-element-in-linq

我想在.Net 4.0中创建一个通用扩展方法,它将生成列表项的百分比到列表的总和。此列表仅为数字类型。

我的代码是

public static T Percentage<T>(this T array) where T: ? // What should come in place of ? 
{
  double[] darray = (double[])T; // Error convert type 'T' to 'double[]'

  darray = darray.Select(x=> x * 100 / darray.Sum()).ToArray();

  return darray; // Error convert type 'double[]' to 'T'
}

我想称之为

double[] MD = {8.0,21.0,25.0,13.0,26.0,37.0,37.0,33.0,71.0,9.0};

MD = MD.Percentage().ToArray(); 

我做错了什么;

3 个答案:

答案 0 :(得分:4)

不可能有你想要的类型约束。查看Constraining a generic type argument to numeric types了解详情。

由于没有太多数字类型(int / float / double / decimal),因此简单地提供具体实现可能更容易。

更多选项

答案 1 :(得分:4)

您无法通过“数字”值约束类型。这在某种程度上是对框架的限制。

你最好的方法就是这样做:

public static IEnumerable<double> Percentage(this IEnumerable<int> values)
{
    var sum = values.Sum();
    return values.Select(v => (double)v / (double)sum).ToArray();
}

public static IEnumerable<double> Percentage(this IEnumerable<long> values)
{
    var sum = values.Sum();
    return values.Select(v => (double)v / (double)sum).ToArray();
}

public static IEnumerable<double> Percentage(this IEnumerable<double> values)
{
    var sum = values.Sum();
    return values.Select(v => v / sum).ToArray();
}

基本上,您需要为要支持的每种类型生成重载。

答案 2 :(得分:1)

有一种方法可行,即传递选择器功能,允许您在调用Percentage时转换为双精度。这是扩展方法:

public static double[] Percentage<T>(this T[] array, Func<T,double> selector)
{
    var doubles = array.Select(selector);
    var sum = doubles.Sum();

    var result = doubles.Select(x => x * 100 / sum).ToArray();

    return result;
}

用法:

    double[] MD = {8.0,21.0,25.0,13.0,26.0,37.0,37.0,33.0,71.0,9.0};
var MDPercent = MD.Percentage(x => x).ToArray();

然后你可以有一个简单的双版本:

public static double[] Percentage(this double[] array)
{
    return Percentage(array, x => x);
}

或者是int版本:

public static double[] Percentage(this int[] array)
{
    return array.Percentage(x => (double)x);
}

哪个更易于使用:

var MDPercent = MD.Percentage().ToArray();