代码之家  ›  专栏  ›  技术社区  ›  KrisTrip

在C#[关闭]中生成随机浮点的最佳方法

  •  49
  • KrisTrip  · 技术社区  · 14 年前

    在C#中生成随机浮点的最佳方法是什么?

    更新:我想要从float.Minvalue到float.Maxvalue的随机浮点数。我在一些数学方法的单元测试中使用这些数字。

    7 回复  |  直到 12 年前
        1
  •  69
  •   user7116    6 年前

    最好的方法,没有疯狂的价值观, distributed with respect to the representable intervals on the floating-point number line

    static float NextFloat(Random random)
    {
        double mantissa = (random.NextDouble() * 2.0) - 1.0;
        // choose -149 instead of -126 to also generate subnormal floats (*)
        double exponent = Math.Pow(2.0, random.Next(-126, 128));
        return (float)(mantissa * exponent);
    }
    

    (*) ... 检查 here 对于次正常浮动

    另一种方法将为您提供一些疯狂的值(位模式的均匀分布),可能对模糊化有用:

    static float NextFloat(Random random)
    {
        var buffer = new byte[4];
        random.NextBytes(buffer);
        return BitConverter.ToSingle(buffer,0);
    }
    

    public static float Generate(Random prng)
    {
        var sign = prng.Next(2);
        var exponent = prng.Next((1 << 8) - 1); // do not generate 0xFF (infinities and NaN)
        var mantissa = prng.Next(1 << 23);
    
        var bits = (sign << 31) + (exponent << 23) + mantissa;
        return IntBitsToFloat(bits);
    }
    
    private static float IntBitsToFloat(int bits)
    {
        unsafe
        {
            return *(float*) &bits;
        }
    }
    

    最不有用的方法:

    static float NextFloat(Random random)
    {
        // Not a uniform distribution w.r.t. the binary floating-point number line
        // which makes sense given that NextDouble is uniform from 0.0 to 1.0.
        // Uniform w.r.t. a continuous number line.
        //
        // The range produced by this method is 6.8e38.
        //
        // Therefore if NextDouble produces values in the range of 0.0 to 0.1
        // 10% of the time, we will only produce numbers less than 1e38 about
        // 10% of the time, which does not make sense.
        var result = (random.NextDouble()
                      * (Single.MaxValue - (double)Single.MinValue))
                      + Single.MinValue;
        return (float)result;
    }
    

    Intel Architecture Software Developer's Manual Volume 1: Basic Architecture. Y轴是对数的(以2为底),因为连续的二进制浮点数之间不存在线性差异。

    Comparison of distributions, logarithmic Y-axis

        2
  •  26
  •   Jon Skeet    14 年前

    Random.NextDouble 然后投出 float ? 这将给你一个介于0和1之间的浮动。

    Random 不应该用于诸如财务或安全之类的敏感问题,并且通常应该在整个应用程序中重用现有实例,或者每个线程重用一个实例(如图所示) 不是线程安全的)。

    float.MinValue , float.MaxValue :

    // Perform arithmetic in double type to avoid overflowing
    double range = (double) float.MaxValue - (double) float.MinValue;
    double sample = rng.NextDouble();
    double scaled = (sample * range) + float.MinValue;
    float f = (float) scaled;
    

        3
  •  5
  •   SunsetQuest    9 年前

    再来一个版本(我觉得这个不错)

    static float NextFloat(Random random)
    {
        (float)(float.MaxValue * 2.0 * (rand.NextDouble()-0.5));
    }
    
    //inline version
    float myVal = (float)(float.MaxValue * 2.0 * (rand.NextDouble()-0.5));
    

    我想这个。。。

    • 第二快(见基准测试)
    • 均匀分布

    还有一个版本……(虽然不是很好,但还是发布了)

    static float NextFloat(Random random)
    {
        return float.MaxValue * ((rand.Next() / 1073741824.0f) - 1.0f);
    }
    
    //inline version
    float myVal = (float.MaxValue * ((rand.Next() / 1073741824.0f) - 1.0f));
    

    我想这个。。。

    • 是最快的(见基准测试)
    • 但是,由于Next()是31位随机值,因此它将仅返回2^31个值(50%的相邻值将具有相同的值)

    本页大部分功能的测试:

     Sunsetquest1: min: 3.402823E+38  max: -3.402823E+38 time: 3096ms
     SimonMourier: min: 3.402823E+38  max: -3.402819E+38 time: 14473ms
     AnthonyPegram:min: 3.402823E+38  max: -3.402823E+38 time: 3191ms
     JonSkeet:     min: 3.402823E+38  max: -3.402823E+38 time: 3186ms
     Sixlettervar: min: 1.701405E+38  max: -1.701410E+38 time: 19653ms
     Sunsetquest2: min: 3.402823E+38  max: -3.402823E+38 time: 2930ms
    
        4
  •  3
  •   Anthony Pegram    14 年前

    我采取了与其他人稍有不同的方法

    static float NextFloat(Random random)
    {
        double val = random.NextDouble(); // range 0.0 to 1.0
        val -= 0.5; // expected range now -0.5 to +0.5
        val *= 2; // expected range now -1.0 to +1.0
        return float.MaxValue * (float)val;
    }
    

    这些评论解释了我在做什么。得到下一个double,把这个数字转换成-1和1之间的值,然后乘以 float.MaxValue .

        5
  •  0
  •   Simon Mourier    12 年前

    static float NextFloat(Random random)
    {
        float f;
        do
        {
            byte[] bytes = new byte[4];
            random.NextBytes(bytes);
            f = BitConverter.ToSingle(bytes, 0);
        }
        while (float.IsInfinity(f) || float.IsNaN(f));
        return f;
    }
    
        6
  •  0
  •   Qedized    7 年前

    假设你想得到一个介于5.5和7之间的浮点数,有3个小数。

    float myFloat;
    int myInt;
    System.Random rnd = new System.Random();
    
    void GenerateFloat()
    {
    myInt = rnd.Next(1, 2000);
    myFloat = (myInt / 1000) + 5.5f;
    }
    

    这样你得到的数字总是大于5.5,小于7。

        7
  •  0
  •   Waheed Sattar    7 年前

    我更喜欢使用下面的代码生成一个小数点到第一个小数点。您可以复制粘贴第三行,通过在字符串“combined”中附加该数字,在小数点后添加更多数字。您可以通过将0和9更改为首选值来设置最小值和最大值。

    Random r = new Random();
    string beforePoint = r.Next(0, 9).ToString();//number before decimal point
    string afterPoint = r.Next(0,9).ToString();//1st decimal point
    //string secondDP = r.Next(0, 9).ToString();//2nd decimal point
    string combined = beforePoint+"."+afterPoint;
    decimalNumber= float.Parse(combined);
    Console.WriteLine(decimalNumber);