- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我需要小列表的高斯随机数进行模拟,因此我尝试了以下操作:
import System.Random
seed = 10101
gen = mkStdGen seed
boxMuller mu sigma (r1,r2) = mu + sigma * sqrt (-2 * log r1) * cos (2 * pi * r2)
这只是 Box-Muller 算法 - 给定 [0,1] 区间内的 r1, r2 均匀随机数,它返回一个高斯随机数。
normals 0 g = []
normals n g = take n $ map (boxMuller 0 1) $ pairs $ randoms g
where pairs (x:y:zs) = (x,y):(pairs zs)
因此,每当我需要随机数列表时,我都会使用这个 normals
函数。
问题一定很明显:它总是生成相同的序列,因为我总是使用相同的种子!我没有得到新的序列,我一直只得到序列的前 n 个值。
当我输入时,我明显假装的是:
x = normal 10
y = normal 50
我将 x 作为 map (boxMuller 0 1) $pairs $randoms g
的前 10 个值,将 y 作为此列表中接下来的 50 个值,依此类推。
当然这是不可能的,因为给定相同的输入,函数必须始终返回相同的值。如何摆脱这个陷阱?
最佳答案
我认为,在抽象生成器的 monad 中进行需要随机数的计算是最干净的事情。该代码如下所示:
我们将把 StdGen 实例放入状态 monad 中,然后在状态 monad 的 get 和 set 方法上提供一些糖,以给我们随机数。
首先,加载模块:
import Control.Monad.State (State, evalState, get, put)
import System.Random (StdGen, mkStdGen, random)
import Control.Applicative ((<$>))
(通常我可能不会指定导入,但这可以很容易地理解每个函数的来源。)
然后我们将定义“需要随机数的计算”monad;在本例中,State StdGen
的别名称为 R
。 (因为“随机”和“兰德”已经意味着其他含义。)
type R a = State StdGen a
R 的工作方式是:定义一个需要随机数流(单子(monad)“ Action ”)的计算,然后使用 runRandom
“运行它”:
runRandom :: R a -> Int -> a
runRandom action seed = evalState action $ mkStdGen seed
这需要一个操作和一个种子,并返回操作的结果。就像通常的evalState
、runReader
等
现在我们只需要 State monad 周围的糖。我们使用get
来获取StdGen,并使用put
来安装新状态。这意味着,要获得一个随机数,我们可以这样写:
rand :: R Double
rand = do
gen <- get
let (r, gen') = random gen
put gen'
return r
我们获取随机数生成器的当前状态,用它来获取新的随机数和新的生成器,保存随机数,安装新的生成器状态,然后返回随机数。
这是一个可以使用 runRandom 运行的“操作”,所以让我们尝试一下:
ghci> runRandom rand 42
0.11040701265689151
it :: Double
这是一个纯函数,因此如果您使用相同的参数再次运行它,您将得到相同的结果。杂质保留在您传递给 runRandom 的“ Action ”内。
无论如何,您的函数需要随机数对,所以让我们编写一个操作来生成一对随机数:
randPair :: R (Double, Double)
randPair = do
x <- rand
y <- rand
return (x,y)
使用 runRandom 运行此命令,您将看到这对数字中的两个不同数字,正如您所期望的那样。但请注意,您不必为“rand”提供参数;这是因为函数是纯函数,但“rand”是一个 Action ,不需要是纯函数。
现在我们可以实现您的法线
函数。 boxMuller
正如您在上面定义的那样,我只是添加了一个类型签名,这样我就可以了解正在发生的事情,而无需阅读整个函数:
boxMuller :: Double -> Double -> (Double, Double) -> Double
boxMuller mu sigma (r1,r2) = mu + sigma * sqrt (-2 * log r1) * cos (2 * pi * r2)
实现所有辅助函数/操作后,我们终于可以编写 normals
,这是一个 0 个参数的操作,返回一个(延迟生成的)正态分布 double 的无限列表:
normals :: R [Double]
normals = mapM (\_ -> boxMuller 0 1 <$> randPair) $ repeat ()
如果您愿意,您也可以写得更简洁:
oneNormal :: R Double
oneNormal = do
pair <- randPair
return $ boxMuller 0 1 pair
normals :: R [Double]
normals = mapM (\_ -> oneNormal) $ repeat ()
repeat ()
为单子(monad) Action 提供了一个无限的流来调用函数(这也是法线结果无限长的原因)。我最初编写了[1..]
,但我重写了它,从程序文本中删除了无意义的1
。我们不是对整数进行操作,我们只是想要一个无限列表。
总之,就这样吧。要在实际程序中使用它,您只需在 R Action 中执行需要随机法线的工作即可:
someNormals :: Int -> R [Double]
someNormals x = liftM (take x) normals
myAlgorithm :: R [Bool]
myAlgorithm = do
xs <- someNormals 10
ys <- someNormals 10
let xys = zip xs ys
return $ uncurry (<) <$> xys
runRandom myAlgorithm 42
应用一元 Action 编程的常用技术;在 R
中保留尽可能少的应用程序,这样事情就不会太困惑。
哦,还有另一件事:惰性可以干净地“泄漏”到 monad 边界之外。这意味着你可以写:
take 10 $ runRandom normals 42
它会起作用。
关于haskell - Haskell 中的随机数采样序列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2110535/
我想做的是让 JTextPane 在 JPanel 中占用尽可能多的空间。对于我使用的 UpdateInfoPanel: public class UpdateInfoPanel extends JP
我在 JPanel 中有一个 JTextArea,我想将其与 JScrollPane 一起使用。我正在使用 GridBagLayout。当我运行它时,框架似乎为 JScrollPane 腾出了空间,但
我想在 xcode 中实现以下功能。 我有一个 View Controller 。在这个 UIViewController 中,我有一个 UITabBar。它们下面是一个 UIView。将 UITab
有谁知道Firebird 2.5有没有类似于SQL中“STUFF”函数的功能? 我有一个包含父用户记录的表,另一个表包含与父相关的子用户记录。我希望能够提取用户拥有的“ROLES”的逗号分隔字符串,而
我想使用 JSON 作为 mirth channel 的输入和输出,例如详细信息保存在数据库中或创建 HL7 消息。 简而言之,输入为 JSON 解析它并输出为任何格式。 最佳答案 var objec
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个以前可能被问过的问题,但我很难找到正确的描述。我希望有人能帮助我。 在下面的代码中,我设置了varprice,我想添加javascript变量accu_id以通过rails在我的数据库中查找记
我有一个简单的 SVG 文件,在 Firefox 中可以正常查看 - 它的一些包装文本使用 foreignObject 包含一些 HTML - 文本包装在 div 中:
所以我正在为学校编写一个 Ruby 程序,如果某个值是 1 或 3,则将 bool 值更改为 true,如果是 0 或 2,则更改为 false。由于我有 Java 背景,所以我认为这段代码应该有效:
我做了什么: 我在这些账户之间创建了 VPC 对等连接 互联网网关也连接到每个 VPC 还配置了路由表(以允许来自双方的流量) 情况1: 当这两个 VPC 在同一个账户中时,我成功测试了从另一个 La
我有一个名为 contacts 的表: user_id contact_id 10294 10295 10294 10293 10293 10294 102
我正在使用 Magento 中的新模板。为避免重复代码,我想为每个产品预览使用相同的子模板。 特别是我做了这样一个展示: $products = Mage::getModel('catalog/pro
“for”是否总是检查协议(protocol)中定义的每个函数中第一个参数的类型? 编辑(改写): 当协议(protocol)方法只有一个参数时,根据该单个参数的类型(直接或任意)找到实现。当协议(p
我想从我的 PHP 代码中调用 JavaScript 函数。我通过使用以下方法实现了这一点: echo ' drawChart($id); '; 这工作正常,但我想从我的 PHP 代码中获取数据,我使
这个问题已经有答案了: Event binding on dynamically created elements? (23 个回答) 已关闭 5 年前。 我有一个动态表单,我想在其中附加一些其他 h
我正在尝试找到一种解决方案,以在 componentDidMount 中的映射项上使用 setState。 我正在使用 GraphQL连同 Gatsby返回许多 data 项目,但要求在特定的 pat
我在 ScrollView 中有一个 View 。只要用户按住该 View ,我想每 80 毫秒调用一次方法。这是我已经实现的: final Runnable vibrate = new Runnab
我用 jni 开发了一个 android 应用程序。我在 GetStringUTFChars 的 dvmDecodeIndirectRef 中得到了一个 dvmabort。我只中止了一次。 为什么会这
当我到达我的 Activity 时,我调用 FragmentPagerAdapter 来处理我的不同选项卡。在我的一个选项卡中,我想显示一个 RecyclerView,但他从未出现过,有了断点,我看到
当我按下 Activity 中的按钮时,会弹出一个 DialogFragment。在对话框 fragment 中,有一个看起来像普通 ListView 的 RecyclerView。 我想要的行为是当
我是一名优秀的程序员,十分优秀!