zoukankan      html  css  js  c++  java
  • 粒子群算法简介及应用

    简介

    定义

    粒子群优化算法(Particle Swarm optimization,PSO)又翻译为粒子群算法、微粒群算法、或微粒群优化算法。是通过模拟鸟群觅食行为而发展起来的一种基于群体协作的随机搜索算法。通常认为它是群集智能 (Swarm intelligence, SI) 的一种。它可以被纳入多主体优化系统(Multiagent Optimization System, MAOS).

    模拟捕食

    SO模拟鸟群的捕食行为。一群鸟在随机搜索食物,在这个区域里只有一块食物。所有的鸟都不知道食物在那里。但是他们知道当前的位置离食物还有多远。那么找到食物的最优策略是什么呢。最简单有效的就是搜寻离食物最近的鸟的周围区域。

    抽象成粒子 

    PSO中,每个优化问题的解都是搜索空间中的一只鸟。我们称之为“粒子”。所有的粒子都有一个由被优化的函数决定的适应值(fitnessvalue),每个粒子还有一个速度决定他们飞翔的方向和距离。然后粒子们就追随当前的最优粒子在解空间中搜索。

    更新

    PSO初始化为一群随机粒子(随机解),然后通过迭代找到最优解,在每一次叠代中,粒子通过跟踪两个“极值”来更新自己。第一个就是粒子本身所找到的最优解,这个解叫做个体极值pBest,另一个极值是整个种群找到的最优解,这个极值是全局极值gBest。另外也可以不用整个种群而只是用其中一部分最优粒子的邻居,那么在所有邻居中的极值就是局部极值。

    在找到这两个最优值时,粒子根据如下的公式来更新自己的速度和新的位置:

    再来看一个公式:

    公式二和公式三被视为标准PSO算法。

    学习因子c1、c2分析

    公式(2)和(3)中pbest和gbest分别表示微粒群的局部和全局最优位置。

    1)当C1=0时,则粒子没有了认知能力,变为只有社会的模型(social-only):

    称为全局PSO算法。粒子有扩展搜索空间的能力,具有较快的收敛速度,但由于缺少局部搜索,对于复杂问题比标准PSO 更易陷入局部最优。

    2)当C2=0时,则粒子之间没有社会信息,模型变为只有认知(cognition-only)模型

    称为局部PSO算法。由于个体之间没有信息的交流,整个群体相当于多个粒子进行盲目的随机搜索,收敛速度慢,因而得到最优解的可能性小。

    标准PSO算法流程

    1)初始化一群微粒(群体规模为N),包括随机位置和速度;

    2)评价每个微粒的适应度;

    3)对每个微粒,将其适应值与其经过的最好位置pbest作比较,如果较好,则将其作为当前的最好位置pbest;

    4)对每个微粒,将其适应值与其经过的最好位置gbest作比较,如果较好,则将其作为当前的最好位置gbest;

    5)根据公式(2)、(3)调整微粒速度和位置;

    6)未达到结束条件则转第2步。

    迭代终止条件根据具体问题一般选为最大迭代次数Gk或(和)微粒群迄今为止搜索到的最优位置满足预定最小适应阈值。

    应用举例

    这里用一个简单的例子说明PSO训练神经网络的过程。这个例子使用分类问题的基准函数 (Benchmark function)IRIS数据集。(Iris 是一种鸢尾属植物) 在数据记录中,每组数据包含Iris花的四种属性:萼片长度,萼片宽度,花瓣长度,和花瓣宽度,三种不同的花各有50组数据. 这样总共有150组数据或模式。

    我们用3层的神经网络来做分类。有四个输入和三个输出。所以神经网络的输入层有4个节点,输出层有3个节点我们也可以动态调节隐含层节点的数目,不过这里我们假定隐含层有6个节点。我们也可以训练神经网络中其他的参数。不过这里我们只是来确定网络权重。粒子就表示神经网络的一组权重,应该是4*6+6*3=42个参数。权重的范围设定为[-100,100] (这只是一个例子,在实际情况中可能需要试验调整).在完成编码以后,我们需要确定适应函数。对于分类问题,我们把所有的数据送入神经网络,网络的权重有粒子的参数决定。然后记录所有的错误分类的数目作为那个粒子的适应值。我们就利用PSO来训练神经网络来获得尽可能低的错误分类数目。PSO本身并没有很多的参数需要调整。所以在实验中只需要调整隐含层的节点数目和权重的范围以取得较好的分类效果。

    参数设置

    PSO中并没有许多需要调节的参数,下面列出了这些参数以及经验设置。

    • 粒子数: 一般取 20–40. 其实对于大部分的问题10个粒子已经足够可以取得好的结果, 不过对于比较难的问题或者特定类别的问题, 粒子数可以取到100 或 200
    • 粒子的长度: 这是由优化问题决定, 就是问题解的长度
    • 粒子的范围: 由优化问题决定,每一维可是设定不同的范围
    • Vmax: 最大速度,决定粒子在一个循环中最大的移动距离,通常设定为粒子的范围宽度,例如上面的例子里,粒子 (x1, x2, x3) x1 属于 [-10, 10], 那么 Vmax 的大小就是 20
    • 学习因子: c1 和 c2 通常等于 2. 不过在文献中也有其他的取值. 但是一般 c1 等于 c2 并且范围在0和4之间
    • 中止条件: 最大循环数以及最小错误要求. 例如, 在上面的神经网络训练例子中, 最小错误可以设定为1个错误分类, 最大循环设定为2000, 这个中止条件由具体的问题确定.
    • 全局PSO和局部PSO: 我们介绍了两种版本的粒子群优化算法: 全局版和局部版. 前者速度快不过有时会陷入局部最优. 后者收敛速度慢一点不过很难陷入局部最优. 在实际应用中, 可以先用全局PSO找到大致的结果,再用局部PSO进行搜索.
    • 另外的一个参数是惯性权重, 由Shi 和Eberhart提出, 有兴趣的可以参考他们1998年的论文(题目: A modified particle swarm optimizer)。

    编码实现

    为了简便,这里直接使用scikit-opt库中的PSO算法。

    def demo_func(x):
        x1, x2, x3 = x
        return x1 ** 2 + (x2 - 0.05) ** 2 + x3 ** 2
    
    
    # %% Do PSO
    from sko.PSO import PSO
    
    pso = PSO(func=demo_func, dim=3, pop=40, max_iter=150, lb=[0, -1, 0.5], ub=[1, 1, 1], w=0.8, c1=0.5, c2=0.5)
    pso.run()
    print('best_x is ', pso.gbest_x, 'best_y is', pso.gbest_y)
    
    # %% Plot the result
    import matplotlib.pyplot as plt
    
    plt.plot(pso.gbest_y_hist)
    plt.show()

    在演示一个更直观的例子:

    # Plot particle history as animation
    import numpy as np
    from sko.PSO import PSO
    
    
    def demo_func(x):
        x1, x2 = x
        return x1 ** 2 + (x2 - 0.5) ** 2
    
    
    pso = PSO(func=demo_func, dim=2, pop=20, max_iter=40, lb=[-1, -1], ub=[1, 1])
    pso.record_mode = True
    pso.run()
    print('best_x is ', pso.gbest_x, 'best_y is', pso.gbest_y)
    
    # %% Now Plot the animation
    import matplotlib.pyplot as plt
    from matplotlib.animation import FuncAnimation
    
    record_value = pso.record_value
    X_list, V_list = record_value['X'], record_value['V']
    
    fig, ax = plt.subplots(1, 1)
    ax.set_title('title', loc='center')
    line = ax.plot([], [], 'b.')
    
    X_grid, Y_grid = np.meshgrid(np.linspace(-1.0, 1.0, 40), np.linspace(-1.0, 1.0, 40))
    Z_grid = demo_func((X_grid, Y_grid))
    ax.contour(X_grid, Y_grid, Z_grid, 20)
    
    ax.set_xlim(-1, 1)
    ax.set_ylim(-1, 1)
    
    plt.ion()
    p = plt.show()
    
    
    def update_scatter(frame):
        i, j = frame // 10, frame % 10
        ax.set_title('iter = ' + str(i))
        X_tmp = X_list[i] + V_list[i] * j / 10.0
        plt.setp(line, 'xdata', X_tmp[:, 0], 'ydata', X_tmp[:, 1])
        return line
    
    
    ani = FuncAnimation(fig, update_scatter, blit=True, interval=25, frames=300)
    plt.show()
    
    ani.save('pso.gif', writer='pillow')

    可见,粒子最后聚集在(0.00,0.50)。

    参考链接:

    1. 百度百科-粒子群优化算法

    2. 简书-【算法】粒子群算法Particle Swarm Optimization超详细解析+代码实例讲解

    3. scikit-opt官方文档-PSO部分

  • 相关阅读:
    Josh Christie's blog翻译
    [翻译]自定义提取规则通过索引提取窗体字段
    [翻译]为什么我不能生成更多的负载?
    [翻译]我能从一个Web测试调用另一个Web测试吗?
    [翻译]Web测试记录器
    [翻译]Web测试提取和验证规则
    测试常见错误(1)
    关于VSTE资料的说明
    [翻译]使用自定义IHttpBody类扩展Web测试
    MSDN上终于有了Tester Center
  • 原文地址:https://www.cnblogs.com/lfri/p/12241765.html
Copyright © 2011-2022 走看看