繁体   English   中英

为什么`fst(Random.split gen)`返回的生成器有时会产生与`gen`相同的结果?

[英]Why does the generator returned by `fst (Random.split gen)` sometimes produce the same results as `gen`?

我正在尝试创建列表的随机排列。 我是函数式语言中的随机性新手,还没有完全掌握 monad,但我以我认为应该有效的方式使用了 Random.newStdGen 和 Random.Shuffle.shuffle'。

我遇到的问题是,我得到了很多重复的排列,以至于我似乎错误地使用或错误地理解了用于生成器的拆分 function。

相关功能在这里:

doGenerateInput :: [[Int]] -> System.Random.StdGen -> Int -> Int -> [[Int]]
doGenerateInput acc gen n 0 = acc
doGenerateInput acc gen n k =
  doGenerateInput
    (System.Random.Shuffle.shuffle' [1 .. n] n gen : acc) accumulator
    (fst (System.Random.split gen))
    n
    (k -1)

generateInput :: Int -> Int -> IO [[Int]]
generateInput n k = do
  gen <- System.Random.newStdGen
  return (doGenerateInput [] gen n k)

这里generateInput应该创建[1..n]k个随机排列。 它在将生成器传递到下一个递归级别之前拆分生成器,因此每个排列应该在统计上彼此无关。 然而,我得到的实际结果包括大量重复。 通常连续两次返回相同的排列。 有时甚至连续三次。 有人对我可能做错的事情有任何建议吗?

这是我从它那里收到的 output。

8 18 1 6 17 19 7 9 15 2 11 12 20 16 10 5 4 14 13 3
12 11 8 20 1 6 19 7 9 17 2 13 14 18 10 5 4 16 15 3
3 13 12 9 1 2 8 10 11 19 4 15 16 20 14 7 6 18 17 5
3 13 12 9 1 2 8 10 11 19 4 15 16 20 14 7 6 18 17 5
7 8 3 15 14 16 20 11 1 2 10 12 13 5 4 19 6 18 17 9
7 8 3 15 14 16 20 11 1 2 10 12 13 5 4 19 6 18 17 9
7 8 3 15 14 16 20 11 1 2 10 12 13 5 4 19 6 18 17 9
7 8 3 15 14 16 20 11 1 2 10 12 13 5 4 19 6 18 17 9
8 6 9 10 7 11 3 19 18 20 15 1 2 14 17 16 4 12 5 13
18 8 6 9 10 7 11 3 20 19 15 1 2 14 17 16 4 12 5 13
8 11 5 13 15 4 20 18 16 14 10 3 12 1 19 7 9 6 2 17
16 8 11 5 13 6 15 18 4 19 17 12 3 14 1 9 10 7 2 20
2 17 9 12 6 14 7 16 19 5 20 18 13 4 15 1 8 11 3 10
7 2 18 10 6 14 8 16 9 19 5 20 15 4 17 1 11 13 3 12
17 7 2 19 13 10 6 15 8 18 9 5 14 11 16 3 1 20 4 12
11 4 19 8 2 6 16 13 9 18 10 12 20 3 7 15 5 1 17 14
17 11 4 20 13 10 8 2 6 19 15 9 3 14 5 18 16 7 1 12
17 11 4 20 13 10 8 2 6 19 15 9 3 14 5 18 16 7 1 12
12 18 11 4 1 15 13 9 3 7 17 10 5 16 6 20 19 8 2 14
12 18 11 4 1 15 13 9 3 7 17 10 5 16 6 20 19 8 2 14

根据split的文档,我希望从 split 返回的两个生成器不相关。 但是重复率似乎表明使用fst (split gen)生成随机数产生与gen相同的结果大约有一半的时间。

返回两个不同的伪随机数生成器。 实现应注意确保生成的生成器不相关。

https://hackage.haskell.org/package/random-1.2.1/docs/System-Random.html#v:split

我找到了解决方案,但我不明白为什么它有效

如果我使用snd (split gen)而不是fst (split gen) ,我不会得到任何重复。 但是根据文档,我不确定为什么。 它没有记录返回的第一个和第二个生成器之间的区别。

任何见解将不胜感激。

random设计中存在某种假设,即RandomGen值仅使用一次。 当您重复使用它们时,可能会发生奇怪的事情。 我不知道如何RandomGensplit来给你这个结果,但我可以告诉你这是假设你不打算做你正在做的事情。 gen传递给splitshuffle'正在使用它两次。 为您的用例使用split的预期方法是预先调用 split,然后将其返回值之一传递给shuffle'并将另一个传递给递归调用。

您不需要使用split ,这通常只在树状结构中需要。

您可以像这样生成排列:

$ ghci
 GHCi, version 8.8.4: https://www.haskell.org/ghc/  :? for help
 ...
 λ> 
 λ> import System.Random
 λ> import Control.Monad
 λ> import Control.Monad.Random
 λ> import System.Random.Shuffle
 λ> 
 λ> :type replicateM
 replicateM :: Applicative m => Int -> m a -> m [a]
 λ> 
 λ> :type shuffleM
 shuffleM :: MonadRandom m => [a] -> m [a]
 λ> 
 λ> action n k = replicateM k (shuffleM [1..n])
 λ> 
 λ> :type  action 10 6
 action 10 6 :: (MonadRandom m, Num a, Enum a) => m [[a]]
 λ> 
 λ> randomSeed = 42
 λ> gen0 = mkStdGen randomSeed
 λ> 
 λ> (xss,gen1) = runRand (action 10 6) gen0
 λ> 
 λ> printAsLines zs = mapM_  (putStrLn . show)  zs
 λ> 
 λ> printAsLines xss
 [10,7,4,6,3,9,1,8,2,5]
 [4,1,3,2,7,8,10,9,6,5]
 [2,8,6,9,1,5,7,4,3,10]
 [7,10,5,2,9,1,6,4,8,3]
 [3,7,4,10,8,1,2,5,6,9]
 [9,1,2,4,3,8,7,6,5,10]
 λ> 

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM