6

我需要一个用于课程中分配的汇编程序的伪随机数生成器算法,我更喜欢一个简单的算法。但是,我不能使用外部库。

什么是用于组装的好的、简单的伪随机数生成器算法?

4

9 回答 9

8

简单的方法是只选择两个大的相对素数 a 和 b,然后继续将随机数乘以 a 并添加 b。使用模运算符将低位保留为随机数,并为下一次迭代保留完整值。

该算法被称为线性同余生成器

于 2008-09-18T05:12:15.543 回答
3

The Art of Computer Programming第 2 卷有很多关于伪随机数生成的信息。这些算法在汇编程序中进行了演示,因此您可以自己查看汇编程序中最简单的算法。

但是,如果您可以链接到外部库或目标文件,那将是您最好的选择。然后你可以链接到,例如,Mersenne Twister

请注意,大多数伪随机数生成器对于密码学来说是不安全的,因此如果您需要安全的随机数生成,您需要超越基本算法(并且可能应该利用特定于操作系统的加密 API)。

于 2008-09-18T05:23:20.570 回答
3

用于测试的简单代码,不要与 Crypto 一起使用

来自测试计算机软件,第 138 页

是 32 位数学,你不需要操作 MOD 2^32

RNG = (69069*RNG + 69069) MOD 2^32
于 2008-09-18T13:32:49.177 回答
2

好吧 - 由于我没有看到对旧的线性反馈移位寄存器的引用,我发布了一些基于 SSE 内在 C 代码。只是为了完整性。几个月前我写了那件事来再次提高我的 SSE 技能。

#include <emmintrin.h>

static __m128i LFSR;

void InitRandom (int Seed)
{
  LFSR = _mm_cvtsi32_si128 (Seed);
}

int GetRandom (int NumBits)
{
  __m128i seed = LFSR;
  __m128i one  = _mm_cvtsi32_si128(1);
  __m128i mask; 
  int i;

  for (i=0; i<NumBits; i++)
  {

    // generate xor of adjecting bits
    __m128i temp = _mm_xor_si128(seed, _mm_srli_epi64(seed,1));

    // generate xor of feedback bits 5,6 and 62,61
    __m128i NewBit = _mm_xor_si128( _mm_srli_epi64(temp,5),
                                    _mm_srli_epi64(temp,61));

    // Mask out single bit: 
    NewBit = _mm_and_si128 (NewBit, one);

    // Shift & insert new result bit:
    seed = _mm_or_si128 (NewBit, _mm_add_epi64 (seed,seed));
  }

  // Write back seed...
  LFSR = seed;

  // generate mask of NumBit ones.
  mask = _mm_srli_epi64 (_mm_cmpeq_epi8(seed, seed), 64-NumBits);

  // return random number:
  return _mm_cvtsi128_si32 (_mm_and_si128(seed,mask));
}

将此代码翻译成汇编程序是微不足道的。只需用真正的 SSE 指令替换内在函数并在其周围添加一个循环。

顺便说一句 - 此代码生成的序列在 4.61169E+18 个数字之后重复。这比您通过主要方法和 32 位算术获得的要多得多。如果展开它也会更快。

于 2008-09-19T18:47:50.463 回答
1

@jjrv
您所描述的实际上是一个线性同群生成器。最随机位是最高位。要从 0..N-1 中获得一个数字,请将完整值乘以 N(32 位乘以 32 位,得到 64 位)并使用高 32 位。

您不应该只为a使用任何数字(从一个完整值到下一个完整值的乘数),Knuth 中推荐的数字(表 1 第 3.3.4 节 TAOCP vol 2 1981)是 1812433253、1566083941、69069 和 1664525。

您可以为b 选择任何奇数。(补充)。

于 2008-09-18T20:21:31.800 回答
1

为什么不使用外部库???那个轮子已经发明了几百次了,为什么还要再做一次呢?

如果您需要自己实现 RNG,您是否需要按需生成数字 - 即您是否正在实现 rand() 函数 - 或者您是否需要生成随机数流 - 例如用于内存测试?

你需要一个加密强度的 RNG 吗?在它重复之前需要多长时间?您是否必须绝对、肯定地保证所有位的均匀分布?

这是我几年前使用的简单技巧。我在嵌入式工作,我需要在上电时测试 RAM,我想要非常小的、快速的代码和非常少的状态,我这样做了:

  • 从种子的任意 4 字节常量开始。
  • 计算这 4 个字节的 32 位 CRC。这给了你接下来的 4 个字节
  • 将这 4 个字节反馈到 CRC32 算法中,就好像它们已被附加一样。这 8 个字节的 CRC32 是下一个值。
  • 只要你愿意,就重复。

这需要很少的代码(尽管您需要一个用于 crc32 函数的表)并且具有很少的状态,但是伪随机输出流在重复之前有很长的循环时间。此外,它不需要处理器上的 SSE。假设您手边有 CRC32 函数,那么实现起来很简单。

于 2008-11-25T21:25:47.493 回答
1

使用masm615编译:

delay_function macro
    mov cx,0ffffh
.repeat
    push cx
    mov cx,0f00h
    .repeat
        dec  cx
        .until cx==0
    pop cx
    dec cx
    .until cx==0
endm

random_num macro
   mov  cx,64    ;assum we want to get 64 random numbers
   mov  si,0

get_num:    
   push cx
   delay_function    ;since cpu clock is fast,so we use delay_function
   mov  ah,2ch  
   int  21h
   mov  ax,dx     ;get clock 1/100 sec
   div  num       ;assume we want to get a number from 0~num-1
   mov  arry[si],ah   ;save to array you set
   inc  si
   pop  cx
   loop get_num   ;here we finish the get_random number 
于 2011-11-17T06:53:54.583 回答
0

您也可以使用不同位之间的 XOR 和元素来模拟移位寄存器,这将为您提供伪随机数字序列。

于 2008-09-18T05:22:40.760 回答
0

线性同余 (X = AX+C mod M) PRNG 可能是分配给汇编程序课程的好方法,因为您的学生必须处理 2^31 以上的中间 AX 结果的进位并计算模数。如果您是学生,它们在汇编程序中实现起来相当简单,并且可能是讲师的想法。

于 2008-09-18T16:51:10.667 回答