热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

【优化算法】粒子群优化算法简介

这里是引用文章目录1.简介2.涌现复杂性3.鸟群智能建模4.代码实现5.Conclusion参考资料1.简介人工智能是计算机科学的一个大领域,它模拟计算机中的智能行

这里是引用





文章目录

  • 1. 简介
  • 2. 涌现复杂性
  • 3. 鸟群智能建模
  • 4. 代码实现
  • 5. Conclusion
  • 参考资料




1. 简介

人工智能是计算机科学的一个大领域,它模拟计算机中的智能行为。在此基础上,提出了一种基于元启发式( metaheuristic)的粒子群优化算法来模拟鸟类觅食、鱼群移动等。这种算法能够模拟群体的行为,以便迭代地优化数值问题。例如,它可以被分类为像蚁群算法、人工蜂群算法和细菌觅食这样的群体智能算法。

J. Kennedy 和 R.Eberhart 在1995年提出的粒子群优化(Particle Swarm Optimization,PSO)变得非常流行,它是一种基于随机优化(Stochastic Optimization)的强大算法,受鸟群中的规则启发,连续优化过程允许多目标和更多的变化。该方法包括不断寻找最佳解,以每次迭代计算的一定速度移动粒子(在这种情况下表示为位置 (x,y)(x,y)(x,y))。每个粒子的运动都有其自身的影响,最著名的位置也是空间搜索中最著名的位置。最终期望的结果是粒子群收敛到最优解。重要的是要提到粒子群算法不使用梯度下降,所以它可以用于非线性问题,只要它不要求问题必须是可微的。

C++/Python代码可参考该仓库。


2. 涌现复杂性

涌现复杂性(Emergent Complexity)是一种现象,描述了大群体的各个组成部分如何以相同但更简单的规则一起工作,以创建多样而复杂的系统。有一些自然的复杂行为可以作为涌现的例子。例如,蚂蚁本能地相互交流,以建立一个活的桥梁,在寻找食物来源时最小化交换距离(Video)。鸟类相互跟随,形成更大的群体,这增加了它们发现捕食者和食物来源的可能性。不像通常的复杂性(complexity)概念,它不一定有用,自然复杂性(natural complexity)是一百万年自然选择过程的结果,在这个过程中,能源的使用是增加生存机会的最重要因素。因此,如果同一个问题有两个解决方案,更简单、需要更少能量的方法将因自然选择而存在。这就是为什么大自然建议的解决方案会很简单,但仍然能有效地尽可能减少动物的能量消耗。因此,科学家们分别观察了一群欧椋鸟(starlings,八哥😸)中的每只鸟,发现它们中的每只都采取简单的行动来形成复杂的结构。


  • 每只鸟都有自己的位置和速度;
  • 每只鸟都有自己的视野,可以跟随周围的鸟。这只鸟完全不知道超出此范围的动作。
  • 如果有任何鸟类发现食物,则该群中的所有鸟类都将趋向该方向。

3. 鸟群智能建模

在自然界中,在一个数值问题中,任何boid(类似鸟的物体)的可观察到的邻近区域都被限制在一定的范围内。因此,它可以收敛到一些局部极小值(minima)或鞍点(saddle point),其中梯度(在该情况下对应的是速度)将为0。然而,拥有一个以上的boid可以让鸟群中的所有鸟关注到误差函数的更大范围,并且如果它们中的任何一只在误差方面看到了更好的位置,就不会陷入局部极小值。对上述原则进行数学建模,使群体找到误差函数的全局最小值。

在这里插入图片描述


f(x)= x²+y² 的函数图像

  • 每个boid都有自己的位置和速度。我们可以把速度看作误差函数的偏导数的向量。
  • 每个boid都记录了它所经历过的最佳位置,这在一定程度上影响了boid的当前速度。
  • 这将是整群鸟见过的最好的位置。因此,它会以预定的速率影响所有boid的速度。

通过使用上述规则,我们可以轻松得出将驱动每个boid的数学方程式。
在这里插入图片描述


计算Delta V的向量表示

在这里插入图片描述
在以上等式中:


  • www:表示惯性,决定了boid的当前速度对 ΔV\Delta VΔV 的影响程度;
  • c1,c2c_1, c_2c1,c2:定义了boid和swarm的最佳记录位置(best-recorded position)将如何分别影响 ΔV\Delta VΔV;

w,c1,c2,learningRatew, c_1, c_2, learningRatew,c1,c2,learningRate 是在优化过程中应微调的超参数。



粒子群优化算法伪代码:
在这里插入图片描述
其中:


  • Vi(k+1)V_i(k+1)Vi(k+1) 是下一个迭代速度;
  • WWW 是惯性参数。该参数影响最后速度值给定的运动传播。
  • C1,C2C_1, C_2C1,C2 是加速度系数。 C1C_1C1 赋予个体最佳值的重要性,而 C2C_2C2 则代表群体最佳值的重要性。
  • PiP_iPi是个体最佳位置,PgP_gPg 是群体最佳位置。在方程式中,衡量了每个参数到粒子实际位置的距离。
  • rand1rand_1rand1rand2rand_2rand2 是随机数,其中 0≤rand≤10≤rand≤10rand1,它们控制每个值的影响:群体和个体,如下所示。
    在这里插入图片描述



4. 代码实现

使用Python来实现对群体智能背(swarm intelligence)后的数学理论进行了讨论和建模。为了测试算法,Rastrigin函数将被用作误差函数,这是优化问题中最具挑战性的函数之一。在平面上有很多余弦振荡会引入无数的局部极小值,在这些极小值中,boid会卡住。(图自:Source)
在这里插入图片描述

在这里插入图片描述

import numpy as npdef error(current_pos):x = current_pos[0]y = current_pos[1]return (x ** 2 - 10 * np.cos(2 * np.pi * x)) + \(y ** 2 - 10 * np.cos(2 * np.pi * y)) + 20def grad_error(current_pos):x = current_pos[0]y = current_pos[1]return np.array([2 * x + 10 * 2 * np.pi * x * np.sin(2 * np.pi * x),2 * y + 10 * 2 * np.pi * y * np.sin(2 * np.pi * y)])

如上所述,每个boid都将具有位置,速度,误差,最佳位置和已知误差(best-known error)。我们还应该编写一个setter函数来在需要时修改参数。

class Particle:def __init__(self, dim, minx, maxx):self.position &#61; np.random.uniform(low&#61;minx, high&#61;maxx, size&#61;dim)self.velocity &#61; np.random.uniform(low&#61;minx, high&#61;maxx, size&#61;dim)self.best_part_pos &#61; self.position.copy()self.error &#61; error(self.position)self.best_part_err &#61; self.error.copy()def setPos(self, pos):self.position &#61; posself.error &#61; error(pos)if self.error < self.best_part_err:self.best_part_err &#61; self.errorself.best_part_pos &#61; pos

粒子群算法类将由误差函数表面上的移动粒子列表组成。要初始化函数&#xff0c;需要函数的维数、boids的数量以及纪元的数量。

class PSO:w &#61; 0.729c1 &#61; 1.49445c2 &#61; 1.49445lr &#61; 0.01def __init__(self, dims, numOfBoids, numOfEpochs):self.swarm_list &#61; [self.Particle(dims, -500, 500) for i in range(numOfBoids)]self.numOfEpochs &#61; numOfEpochsself.best_swarm_position &#61; np.random.uniform(low&#61;-500, high&#61;500, size&#61;dims)self.best_swarm_error &#61; 1e80 # Set high value to best swarm error

最后&#xff0c;编写代码来找到最佳优化误差函数的位置。首先&#xff0c;在每个时期&#xff0c;每个粒子都被逐个挑选并优化其位置。一旦粒子的位置更新&#xff0c;“如果”语句检查它是否是粒子群的最佳位置。

def optimize(self):for i in range(self.numOfEpochs):for j in range(len(self.swarm_list)):current_particle &#61; self.swarm_list[j] # get current particleVcurr &#61; grad_error(current_particle.position) # calculate current velocity of the particledeltaV &#61; self.w * Vcurr \&#43; self.c1 * (current_particle.best_part_pos - current_particle.position) \&#43; self.c2 * (self.best_swarm_position - current_particle.position) # calculate delta Vnew_position &#61; self.swarm_list[j].position - self.lr * deltaV # calculate the new positionself.swarm_list[j].setPos(new_position) # update the position of particleif error(new_position) < self.best_swarm_error: # check the position if it&#39;s best for swarmself.best_swarm_position &#61; new_positionself.best_swarm_error &#61; error(new_position)print(&#39;Epoch: {0} | Best position: [{1},{2}] | Best known error: {3}&#39;.format(i,self.best_swarm_position[0],self.best_swarm_position[1],self.best_swarm_error))

运行PSO并观察算法的性能。

pso &#61; PSO(dims&#61;2, numOfBoids&#61;30, numOfEpochs&#61;500)
pso.optimize()

...
Epoch: 27 | Best position: [2.1199768747964054,-1.1332450655810595] | Best known error: 11.792438949384607
Epoch: 28 | Best position: [2.1199768747964054,-1.1332450655810595] | Best known error: 11.792438949384607
Epoch: 29 | Best position: [2.1199768747964054,-1.1332450655810595] | Best known error: 11.792438949384607
Epoch: 30 | Best position: [2.1199768747964054,-1.1332450655810595] | Best known error: 11.792438949384607
Epoch: 31 | Best position: [2.1199768747964054,-1.1332450655810595] | Best known error: 11.792438949384607
Epoch: 32 | Best position: [1.0839903018536725,-1.1678886593978168] | Best known error: 8.966098069909691

从下面的等高线图可以很容易地看出&#xff0c;swarm只需要几个纪元就能收敛到Rastrigin函数的全局最小值。要获得创建轮廓可视化的代码以及过程的错误时期图&#xff0c;可以参考&#xff1a;T. Ahadli, Particle Swarm Optimization C&#43;&#43;/Python Project Codes。

在这里插入图片描述
完整代码


"""
Copyright © 2020 Tarlan Ahadli
"""
import numpy as np
import matplotlib.pyplot as pltdef error(current_pos):x &#61; current_pos[0]y &#61; current_pos[1]return (x ** 2 - 10 * np.cos(2 * np.pi * x)) &#43; \(y ** 2 - 10 * np.cos(2 * np.pi * y)) &#43; 20def grad_error(current_pos):x &#61; current_pos[0]y &#61; current_pos[1]return np.array([2 * x &#43; 10 * 2 * np.pi * x * np.sin(2 * np.pi * x),2 * y &#43; 10 * 2 * np.pi * y * np.sin(2 * np.pi * y)])class PSO:class Particle:def __init__(self, dim, minx, maxx):self.position &#61; np.random.uniform(low&#61;minx, high&#61;maxx, size&#61;dim)self.velocity &#61; np.random.uniform(low&#61;minx, high&#61;maxx, size&#61;dim)self.best_part_pos &#61; self.position.copy()self.error &#61; error(self.position)self.best_part_err &#61; self.error.copy()def setPos(self, pos):self.position &#61; posself.error &#61; error(pos)if self.error < self.best_part_err:self.best_part_err &#61; self.errorself.best_part_pos &#61; posw &#61; 0.729c1 &#61; 1.49445c2 &#61; 1.49445lr &#61; 0.01def __init__(self, dims, numOfBoids, numOfEpochs):self.swarm_list &#61; [self.Particle(dims, -500, 500) for i in range(numOfBoids)]self.numOfEpochs &#61; numOfEpochsself.best_swarm_position &#61; np.random.uniform(low&#61;-500, high&#61;500, size&#61;dims)self.best_swarm_error &#61; 1e80 # Set high value to best swarm errordef optimize(self):for i in range(self.numOfEpochs):for j in range(len(self.swarm_list)):current_particle &#61; self.swarm_list[j] # get current particleVcurr &#61; grad_error(current_particle.position) # calculate current velocity of the particledeltaV &#61; self.w * Vcurr \&#43; self.c1 * (current_particle.best_part_pos - current_particle.position) \&#43; self.c2 * (self.best_swarm_position - current_particle.position) # calculate delta Vnew_position &#61; self.swarm_list[j].position - self.lr * deltaV # calculate the new positionself.swarm_list[j].setPos(new_position) # update the position of particleif error(new_position) < self.best_swarm_error: # check the position if it&#39;s best for swarmself.best_swarm_position &#61; new_positionself.best_swarm_error &#61; error(new_position)print(&#39;Epoch: {0} | Best position: [{1},{2}] | Best known error: {3}&#39;.format(i,self.best_swarm_position[0],self.best_swarm_position[1],self.best_swarm_error))pso &#61; PSO(dims&#61;2, numOfBoids&#61;30, numOfEpochs&#61;500)
pso.optimize()

5. Conclusion

总而言之&#xff0c;粒子群优化&#xff08;Particle Swarm Optimization&#xff09;模拟了鸟或鱼群的集体行为。它受益于自然界解决自身优化问题的方式&#xff0c;以最大限度地减少能量使用。自然的设计和原理在计算机科学问题上有很多出色的实际应用。




参考资料


Nature-Inspired Optimization Algorithms: Particle Swarm Optimization Using Python
Implementing the Particle Swarm Optimization (PSO) Algorithm in Python



推荐阅读
  • EzPP 0.2发布,新增YAML布局渲染功能
    EzPP发布了0.2.1版本,新增了YAML布局渲染功能,可以将YAML文件渲染为图片,并且可以复用YAML作为模版,通过传递不同参数生成不同的图片。这个功能可以用于绘制Logo、封面或其他图片,让用户不需要安装或卸载Photoshop。文章还提供了一个入门例子,介绍了使用ezpp的基本渲染方法,以及如何使用canvas、text类元素、自定义字体等。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 十大经典排序算法动图演示+Python实现
    本文介绍了十大经典排序算法的原理、演示和Python实现。排序算法分为内部排序和外部排序,常见的内部排序算法有插入排序、希尔排序、选择排序、冒泡排序、归并排序、快速排序、堆排序、基数排序等。文章还解释了时间复杂度和稳定性的概念,并提供了相关的名词解释。 ... [详细]
  • Python使用Pillow包生成验证码图片的方法
    本文介绍了使用Python中的Pillow包生成验证码图片的方法。通过随机生成数字和符号,并添加干扰象素,生成一幅验证码图片。需要配置好Python环境,并安装Pillow库。代码实现包括导入Pillow包和随机模块,定义随机生成字母、数字和字体颜色的函数。 ... [详细]
  • 不同优化算法的比较分析及实验验证
    本文介绍了神经网络优化中常用的优化方法,包括学习率调整和梯度估计修正,并通过实验验证了不同优化算法的效果。实验结果表明,Adam算法在综合考虑学习率调整和梯度估计修正方面表现较好。该研究对于优化神经网络的训练过程具有指导意义。 ... [详细]
  • 基于dlib的人脸68特征点提取(眨眼张嘴检测)python版本
    文章目录引言开发环境和库流程设计张嘴和闭眼的检测引言(1)利用Dlib官方训练好的模型“shape_predictor_68_face_landmarks.dat”进行68个点标定 ... [详细]
  • 开源Keras Faster RCNN模型介绍及代码结构解析
    本文介绍了开源Keras Faster RCNN模型的环境需求和代码结构,包括FasterRCNN源码解析、RPN与classifier定义、data_generators.py文件的功能以及损失计算。同时提供了该模型的开源地址和安装所需的库。 ... [详细]
  • Python实现变声器功能(萝莉音御姐音)的方法及步骤
    本文介绍了使用Python实现变声器功能(萝莉音御姐音)的方法及步骤。首先登录百度AL开发平台,选择语音合成,创建应用并填写应用信息,获取Appid、API Key和Secret Key。然后安装pythonsdk,可以通过pip install baidu-aip或python setup.py install进行安装。最后,书写代码实现变声器功能,使用AipSpeech库进行语音合成,可以设置音量等参数。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • 解决python matplotlib画水平直线的问题
    本文介绍了在使用python的matplotlib库画水平直线时可能遇到的问题,并提供了解决方法。通过导入numpy和matplotlib.pyplot模块,设置绘图对象的宽度和高度,以及使用plot函数绘制水平直线,可以解决该问题。 ... [详细]
  • Day2列表、字典、集合操作详解
    本文详细介绍了列表、字典、集合的操作方法,包括定义列表、访问列表元素、字符串操作、字典操作、集合操作、文件操作、字符编码与转码等内容。内容详实,适合初学者参考。 ... [详细]
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • C语言常量与变量的深入理解及其影响
    本文深入讲解了C语言中常量与变量的概念及其深入实质,强调了对常量和变量的理解对于学习指针等后续内容的重要性。详细介绍了常量的分类和特点,以及变量的定义和分类。同时指出了常量和变量在程序中的作用及其对内存空间的影响,类似于const关键字的只读属性。此外,还提及了常量和变量在实际应用中可能出现的问题,如段错误和野指针。 ... [详细]
  • Android日历提醒软件开源项目分享及使用教程
    本文介绍了一款名为Android日历提醒软件的开源项目,作者分享了该项目的代码和使用教程,并提供了GitHub项目地址。文章详细介绍了该软件的主界面风格、日程信息的分类查看功能,以及添加日程提醒和查看详情的界面。同时,作者还提醒了读者在使用过程中可能遇到的Android6.0权限问题,并提供了解决方法。 ... [详细]
  • Python操作MySQL(pymysql模块)详解及示例代码
    本文介绍了使用Python操作MySQL数据库的方法,详细讲解了pymysql模块的安装和连接MySQL数据库的步骤,并提供了示例代码。内容涵盖了创建表、插入数据、查询数据等操作,帮助读者快速掌握Python操作MySQL的技巧。 ... [详细]
author-avatar
无敌BUG
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有