小白都能看懂的蒙特卡洛方法以及python实现

1.什么是蒙特卡洛方法(Monte Carlo method)

蒙特卡罗方法也称统计模拟方法,是1940年代中期由于科学技术的发展和电子计算机的发明,而提出的一种以概率统计理论为指导的数值计算方法。是指使用随机数(或更常见的伪随机数)来解决很多计算问题的方法。https://www.51design.com/work...
https://www.51design.com/work...
https://www.51design.com/work...
https://www.51design.com/work...
https://www.51design.com/work...
20世纪40年代,在冯·诺伊曼,斯塔尼斯拉夫·乌拉姆和尼古拉斯·梅特罗波利斯在洛斯阿拉莫斯国家实验室为核武器计划工作时,发明了蒙特卡罗方法。因为乌拉姆的叔叔经常在摩纳哥的蒙特卡洛赌场输钱得名,而蒙特卡罗方法正是以概率为基础的方法。
与它对应的是确定性算法。

2.蒙特卡洛方法的基本思想

通常蒙特卡罗方法可以粗略地分成两类:一类是所求解的问题本身具有内在的随机性,借助计算机的运算能力可以直接模拟这种随机的过程。例如在核物理研究中,分析中子在反应堆中的传输过程。中子与原子核作用受到量子力学规律的制约,人们只能知道它们相互作用发生的概率,却无法准确获得中子与原子核作用时的位置以及裂变产生的新中子的行进速率和方向。科学家依据其概率进行随机抽样得到裂变位置、速度和方向,这样模拟大量中子的行为后,经过统计就能获得中子传输的范围,作为反应堆设计的依据。
另一种类型是所求解问题可以转化为某种随机分布的特征数,比如随机事件出现的概率,或者随机变量的期望值。通过随机抽样的方法,以随机事件出现的频率估计其概率,或者以抽样的数字特征估算随机变量的数字特征,并将其作为问题的解。这种方法多用于求解复杂的多维积分问题。

3.蒙特卡洛求定积分

蒙特卡洛方法的一个重要应用就是求定积分。来看下面的一个例子(参考文献2)
小白都能看懂的蒙特卡洛方法以及python实现_第1张图片

当我们在[a,b]之间随机取一点x时,它对应的函数值就是f(x)。接下来我们就可以用f(x) * (b - a)来粗略估计曲线下方的面积,也就是我们需要求的积分值,当然这种估计(或近似)是非常粗略的。
小白都能看懂的蒙特卡洛方法以及python实现_第2张图片

在此图中,做了四次随机采样,得到了四个随机样本x 1 , x 2 , x 3 , x 4 x_1, x_2, x_3, x_4x1​,x2​,x3​,x4​,并且得到了这四个样本的f ( x i ) f(x_i)f(xi​)的值分别为f ( x 1 ) , f ( x 2 ) , f ( x 3 ) , f ( x 4 ) f(x_1), f(x_2), f(x_3), f(x_4)f(x1​),f(x2​),f(x3​),f(x4​)。对于这四个样本,每个样本能求一个近似的面积值,大小为f ( x i ) ∗ ( b − a ) f(x_i)*(b-a)f(xi​)∗(b−a)。为什么能这么干么?对照图下面那部分很容易理解,每个样本都是对原函数f的近似,所以我们认为f ( x ) f(x)f(x)的值一直都等于f ( x i ) f(x_i)f(xi​)。

按照图中的提示,求出上述面积的数学期望,就完成了蒙特卡洛积分。

如果用数学公式表达上述过程:
S = 1 4 ( f ( x 1 ) ( b − a ) + f ( x 2 ) ( b − a ) + f ( x 3 ) ( b − a ) + f ( x 4 ) ( b − a ) ) = 1 4 ( b − a ) ( f ( x 1 ) + f ( x 2 ) + f ( x 3 ) + f ( x 4 ) ) = 1 4 ( b − a ) ∑ i = 1 4 f ( x i )

S​=41​(f(x1​)(b−a)+f(x2​)(b−a)+f(x3​)(b−a)+f(x4​)(b−a))=41​(b−a)(f(x1​)+f(x2​)+f(x3​)+f(x4​))=41​(b−a)i=1∑4​f(xi​)​

对于更一般的情况,假设要计算的积分如下:
I = ∫ a b g ( x ) d x I = int _a ^ b g(x) dxI=∫ab​g(x)dx
其中被积函数g ( x ) g(x)g(x)在[a,b]内可积。如果选择一个概率密度函数为f X ( x ) f_X(x)fX​(x)的方式进行抽样,并且满足∫ a b f X ( x ) d x = 1 int _a ^ b f_X(x)dx = 1∫ab​fX​(x)dx=1,那么令g ∗ ( x ) = g ( x ) f X ( x ) g^*(x) = frac{g(x)}{f_X(x)}g∗(x)=fX​(x)g(x)​,原有的积分可以写成如下形式:
I = ∫ a b g ∗ ( x ) f X ( x ) d x I = int _a ^ b g^*(x) f_X(x)dxI=∫ab​g∗(x)fX​(x)dx

那么我们求积分的步骤应该是:
1.产生服从分布律F X F_XFX​的随机变量X i ( i = 1 , 2 , ⋯   , N ) X_i(i = 1, 2, cdots,N)Xi​(i=1,2,⋯,N)
2.计算均值
I ‾ = 1 N ∑ i = 1 N g ∗ ( X i ) overline I = frac{1}{N} sum_{i = 1}^N g^*(X_i)I=N1​i=1∑N​g∗(Xi​)
此时有 I ‾ ≈ I overline I approx II≈I

当然实际应用中,我们最常用的还是取f X f_XfX​为均匀分布:
f X ( x ) = 1 b − a , a ≤ x ≤ b f_X(x) = frac{1}{b - a}, a le x le bfX​(x)=b−a1​,a≤x≤b
此时
g ∗ ( x ) = ( b − a ) g ( x ) g^*(x) = (b-a)g(x)g∗(x)=(b−a)g(x)
代入积分表达式有:
I = ( b − a ) ∫ a b g ( x ) 1 b − a d x I = (b-a) int_a^b g(x) frac{1}{b-a}dxI=(b−a)∫ab​g(x)b−a1​dx

最后有
I ‾ = b − a N ∑ i = 1 N g ( X i ) overline I = frac{b-a}{N}sum_{i=1}^Ng(X_i)I=Nb−a​i=1∑N​g(Xi​)

如果从直观上理解这个式子也非常简洁明了:
在[a,b]区间上按均匀分布取N个随机样本X i X_iXi​,计算g ( X i ) g(X_i)g(Xi​)并取均值,得到的相当于Y坐标值,然后乘以( b − a ) (b-a)(b−a)为X坐标长度,得到的即为对应矩形的面积,即积分值。

4.蒙特卡洛方法python实例

首先看一个经典的用蒙特卡洛方法求π piπ值。

`import random


def calpai():
    n = 1000000
    r = 1.0
    a, b = (0.0, 0.0)
    x_neg, x_pos = a - r, a + r
    y_neg, y_pos = b - r, b + r

    count = 0
    for i in range(0, n):
        x = random.uniform(x_neg, x_pos)
        y = random.uniform(y_neg, y_pos)
        if x*x + y*y <= 1.0:
            count += 1

    print (count / float(n)) * 4` 

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

简单介绍下思路:
正方形内部有一个相切的圆,它们的面积之比是π/4。现在,在这个正方形内部,随机产生n个点,计算它们与中心点的距离,并且判断是否落在圆的内部。若这些点均匀分布,则圆周率 pi=4 * count/n, 其中count表示落到圆内投点数 n:表示总的投点数。

然后看一个求定积分的例子。
假设我们想求∫ 0 1 x 2 d x int_0^1 x^2 dx∫01​x2dx的值。具体代码如下。

`def integral():
    n = 1000000
    x_min, x_max = 0.0, 1.0
    y_min, y_max = 0.0, 1.0

    count = 0
    for i in range(0, n):
        x = random.uniform(x_min, x_max)
        y = random.uniform(y_min, y_max)
        # x*x > y,表示该点位于曲线的下面。所求的积分值即为曲线下方的面积与正方形面积的比。
        if x*x > y:
            count += 1

    integral_value = count / float(n)
    print integral_value` 

*   1
*   2
*   3
*   4
*   5
*   6
*   7
*   8
*   9
*   10
*   11
*   12
*   13
*   14
*   15

代码运行的结果为:

`0.332665` 

*   1

由此可见,与解析值的结果误差还是比较小的。

你可能感兴趣的:(python)