7

我们有一个非常数据密集的系统。它存储原始数据,然后根据正确响应的数量/总试验计算百分比。

最近我们有客户想要将旧数据导入我们的系统。

我需要一种将百分比转换为最接近的分数的方法。

例子。

  1. 33% 需要给我 2/6。即使 1/3 是 .33333333
  2. 67% 需要给我 4/6。即使 4/6 是 .6666667

我意识到我可以将其计算为 67/100,但这意味着当 6 就足够时,我必须向系统添加 100 个数据点。

有没有人有任何想法?

编辑 分母可以是任何东西。他们给了我一个原始的、四舍五入的百分比,我试图用原始数据尽可能接近它

4

4 回答 4

4

您的要求是矛盾的:一方面,您想要“将百分比转换为最接近的分数”(*),但另一方面,您想要具有小(est)数字的分数。您需要在何时/如何降低精度以支持较小的数字时找到一些折衷方案。你目前的问题是无法解决的。

(*) 任何给定(整数)百分比 n 的最接近分数 f 是 n/100。根据定义。

于 2013-01-14T15:25:34.250 回答
1

我试图通过使用连分数来满足您的要求。通过将深度限制为三个,我得到了一个合理的近似值。

我未能在合理的时间内提出迭代(或递归)方法。不过,我已经清理了一点。(我知道 3 个字母的变量名不好,但我想不出好名字:-/)

该代码为您提供了它可以找到的指定容差内的最佳有理近似值。得到的分数被减少并且是具有相同或更低分母的所有分数中的最佳近似值。

public partial class Form1 : Form
{
    Random rand = new Random();

    public Form1()
    {
        InitializeComponent();
    }

    private void button1_Click(object sender, EventArgs e)
    {
        for (int i = 0; i < 10; i++)
        {
            double value = rand.NextDouble();
            var fraction = getFraction(value);
            var numerator = fraction.Key;
            var denominator = fraction.Value;

            System.Console.WriteLine(string.Format("Value {0:0.0000} approximated by {1}/{2} = {3:0.0000}", value, numerator, denominator, (double)numerator / denominator));
        }
        /*
            Output:
            Value 0,4691 approximated by 8/17 = 0,4706
            Value 0,0740 approximated by 1/14 = 0,0714
            Value 0,7690 approximated by 3/4 = 0,7500
            Value 0,7450 approximated by 3/4 = 0,7500
            Value 0,3748 approximated by 3/8 = 0,3750
            Value 0,7324 approximated by 3/4 = 0,7500
            Value 0,5975 approximated by 3/5 = 0,6000
            Value 0,7544 approximated by 3/4 = 0,7500
            Value 0,7212 approximated by 5/7 = 0,7143
            Value 0,0469 approximated by 1/21 = 0,0476
            Value 0,2755 approximated by 2/7 = 0,2857
            Value 0,8763 approximated by 7/8 = 0,8750
            Value 0,8255 approximated by 5/6 = 0,8333
            Value 0,6170 approximated by 3/5 = 0,6000
            Value 0,3692 approximated by 3/8 = 0,3750
            Value 0,8057 approximated by 4/5 = 0,8000
            Value 0,3928 approximated by 2/5 = 0,4000
            Value 0,0235 approximated by 1/43 = 0,0233
            Value 0,8528 approximated by 6/7 = 0,8571
            Value 0,4536 approximated by 5/11 = 0,4545
         */
    }

    private KeyValuePair<int, int> getFraction(double value, double tolerance = 0.02)
    {
        double f0 = 1 / value;
        double f1 = 1 / (f0 - Math.Truncate(f0));

        int a_t = (int)Math.Truncate(f0);
        int a_r = (int)Math.Round(f0);
        int b_t = (int)Math.Truncate(f1);
        int b_r = (int) Math.Round(f1);
        int c = (int)Math.Round(1 / (f1 - Math.Truncate(f1)));

        if (Math.Abs(1.0 / a_r - value) <= tolerance)
            return new KeyValuePair<int, int>(1, a_r);
        else if (Math.Abs(b_r / (a_t * b_r + 1.0) - value) <= tolerance)
            return new KeyValuePair<int, int>(b_r, a_t * b_r + 1);
        else
            return new KeyValuePair<int, int>(c * b_t + 1, c * a_t * b_t + a_t + c);
    }
}
于 2013-01-14T16:44:26.443 回答
0

在这里回答我自己的问题。这行得通吗?

    public static Fraction Convert(decimal value) {
    for (decimal numerator = 1; numerator <= 10; numerator++) {
        for (decimal denomenator = 1; denomenator < 10; denomenator++) {
            var result = numerator / denomenator;
            if (Math.Abs(value - result) < .01m)
                return new Fraction() { Numerator = numerator, Denomenator = denomenator };
        }
    }

    throw new Exception();
}

这将使我的分母保持在 10 以下。

于 2013-01-14T15:14:53.440 回答
0

它是否必须返回 2/6 而不是 1/3?如果它总是在六分之二,那么

Math.Round((33 * 6)/100) = 2
于 2013-01-14T15:06:03.737 回答