使C#算法更有效

时间:2008-12-04 10:03:49

标签: c# algorithm performance

我有一个C#方法,它将一个数字的值从一个区间投射到一个目标区间 例如:我们的间隔为-1000和9000,值为5000;如果我们想将这个值投影到0..100的间隔,我们得到60。

以下是方法:

/// <summary>  
/// Projects a value to an interval
/// </summary>
/// <param name="val">The value that needs to be projected</param>  
/// <param name="min">The minimum of the interval the value comes from</param>  
/// <param name="max">The maximum of the interval the value comes from</param>  
/// <param name="intervalTop">The minimum of the interval the value will 
/// be projected to</param>  
/// <param name="intervalBottom">The maximum of the interval the value will 
/// be projected to</param>  
/// <returns>Projected value</returns> 
public decimal ProjectValueToInterval(decimal val,  
                                      decimal min,  
                                      decimal max,  
                                      decimal intervalBottom, 
                                      decimal intervalTop)  
{  
    decimal newMin = Math.Min(0, min);
    decimal valueIntervalSize = Math.Abs(max - newMin);
    decimal targetIntervalSize = Math.Abs(intervalTop - intervalBottom);

    decimal projectionUnit = targetIntervalSize / valueIntervalSize;

    return (val * projectionUnit) + Math.Abs((newMin * projectionUnit));
}

需要为数千个值调用此方法 我想知道在C#中是否有更有效的方法来做到这一点?如果是,您建议做出哪些更改?

5 个答案:

答案 0 :(得分:7)

只有数千个价值观?你真的需要进一步优化吗?我无法想象它现在实际上是一个瓶颈。您是否已对该应用进行了分析,以确定这确实是一个问题?

鉴于该方法是O(1),您不会进行通常目标最激烈的优化 - 提高复杂性。

话虽如此 - 当你召唤这几千次时,任何一个值是否保持不变?例如,您是否重复使用相同的最小值和最大值?如果是这样,您可以创建一个类,它在构造函数中获取这些值并预先计算它可以执行的操作,然后使用一个方法获取其余参数。这会略微改善一些事情,但我回到原来的观点 - 只有在实际导致问题时才会担心这一点。

答案 1 :(得分:4)

答案是:不要使用小数进行快速操作。

为什么浮动或双重不适合你?

答案 2 :(得分:4)

只是数学。你所谓的“投射”是范围A-B和A'-B'的标准化,使得:

比率r =(x-A)/(B-A)=(y-A')/(B'-A')

使用您的条款是:

(val-min)/(max-min)=(returnValue-intervalBottom)/(intervalTop-intervalBottom)

将returnValue解析为:

returnValue =  ((intervalTop-intervalBottom) * (val-min) / (max-min)) + intervalBottom

答案 3 :(得分:2)

除了已经建议的不使用Decimal之外,你可以在其他地方放弃你的最大/最小值,这样你就不需要在所有地方进行所有那些Abs调用。

我怀疑只有部分需要重复执行的浮点数乘以浮点数后跟(或继续)。其他所有东西都可以预先检查&amp;预先计算的。

答案 4 :(得分:1)

您的代码看起来比实际需要的更复杂。公式是:

intervalTop + (intervalBottom - intervalTop) * (val - min) / (max - min);

这比你的版本简单得多(适用于整数类型)。那里没有条件分支(Math.Min调用)或方法调用。好吧,它假设intervalTop&lt; intervalBottom和min&lt;最大。如果intervalTop,intervalBottom,min和max对于一组值是常量,那么您可以预先计算(intervalBottom - intervalTop)和(max - min)并将这些结果用于对函数的所有调用。您可以消除的另一个开销是将函数内联到调用循环中。我不知道C#(或者说JIT编译器)对内联方法做了什么,所以这可能已经在幕后发生了。

如果您可以使用整数或浮点数据类型,一种可能的解决方案是使用SIMD,但这意味着编写可能会破坏您的要求的本机汇编代码。