【问题标题】:Scaling Random Bytes to Selected Integer Range将随机字节缩放到选定的整数范围
【发布时间】:2012-11-21 20:58:06
【问题描述】:

我有一个真正随机字节的文件。我想要一个函数,它通过从文件中获取一个字节并对其进行缩放来返回给定范围内的随机整数 。 (这个词对吗?)

public int getInt(int l, int h) throws IOException {
    int m = (h - l) + 1;            // number of ranges needed
    int r = 256 / m;                // size of byte range
    int x = (r * m) - 1;            // maximum allowable byte value
    int b;
    do {
        try {                       // get random byte from file
            b = ram.readUnsignedByte();
        } catch (EOFException e) {  // catch EOF, reset pointer
            b = 255; ram.seek(0);   // and set b to maximum value
        }                           // so test will fail.
    } while(b > x);                 // if byte is greater than
                                    // allowable value, loop.
    return (b / r) + l;             // return random integer
}                                   // within requested range

这就是我的功能。我担心通过缩放来破坏文件中字节的真正随机性。我读到我需要丢弃任何超出允许最大值的数字(因此对于数字 0-9,最大值为 249,因为我只剩下 7 个值可以分配给 10 个不同的组)。我的实现看起来正确吗?

另外,我想知道,仅仅通过使某些太大的字节无效,我是否会以任何方式扭曲分布?

【问题讨论】:

  • 抱歉,错过了为您提供的范围。必须消失,所以我删除了我的答案,因为它没有处理范围。
  • 没关系,不过我喜欢你读入更多字节并将它们转换为更大值的想法。我稍后可能会使用它,但只是想确保这个基本实现不会破坏文件的任何随机性。
  • 一个 32 位整数由四个字节组成。因此,您可以安全地读取 4 个字节并将它们视为一个有符号整数。
  • 我并没有尝试将 4 字节的块视为 int,而是使用单个字节在传递给函数的范围内生成随机数 (int l, int h)
  • @user1684045 明确一点:所以你只想取 一个 字节,然后在这些值之间缩放,如果h-l>255?

标签: java file random numbers


【解决方案1】:

是的,为了避免偏差,你不能使用模数,你必须抛出不在范围内的结果。

编程成功的关键是将您的任务分成合适的子任务。快速规格:

  1. 添加一个函数来计算存储给定数字需要多少位
  2. 添加一个从随机文件中读取和缓冲字节的类,并有方法为您提供一个整数,其中包含从文件中获取的一些位(其余位为 0)。
  3. 添加获取随机数的实际方法:
    • 计算结果的范围,并从中计算所需的位数
    • 循环获取位,添加下限,如果结果超过上限则重试

关于第 2 步的注意事项:第一个实现可能非常粗糙,例如,您可以只获取 4 个字节作为整数并丢弃额外的位。稍后您可以优化此类以保留未使用的位并在下次使用它们,以避免浪费随机位。由于获得真正好的随机位通常有点昂贵,因此这种优化可能值得认真使用。

对于位操作,例如看这个 SO 问题: Java "Bit Shifting" Tutorial?

【讨论】:

  • 好主意,关于如何保留未使用的位有什么建议吗?
  • 没有时间编写完整的示例代码,但最简单的方法是添加成员变量int lastReadByte;int usedBitsInLastReadByte;,并且在一个方法中,您还可以拥有临时变量int unusedBits = 8 - this.usedBitsInLastReadByte;,然后休息就可以了使用当前字节中的所有 8 个字节时操作和读取下一个字节,请参阅我添加到答案本身的链接。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-05-25
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多