PSO算法

简介: 粒子群算法(Particle Swarm Optimization,简称PSO)是一种优化算法,模拟了鸟群或鱼群等群体生物行为的优化思想。

🍎道阻且长,行则将至。🍓


粒子群算法(Particle Swarm Optimization,简称PSO)是一种优化算法,模拟了鸟群或鱼群等群体生物行为的优化思想。
PSO-1(1).gif

其基本思想是将待求解问题看成一个在多维空间中寻找最优解的优化问题,将每个可能的解看成多维空间中的一个粒子,并将它们随机散布在搜索空间中。
粒子的位置表示一个可行解,粒子的速度表示更新时的变化量。通过给每个粒子分配一个随机的速度向量,指导每个粒子进行探索。同时,使用全局最优和局部最优导引粒子的搜索方向。全局最优即全局最优解的位置,局部最优为某个粒子在个体搜索阶段找到的最优解。将这些信息作为粒子的引导方向,逐渐逼近全局最优解,达到最优化的目的。

1.PSO算法主要步骤🌱

  1. 设定粒子群种群大小目标函数搜索空间

初始化粒子群:每个粒子的位置和速度都要设置随机初值,并计算其个体适应度

  1. 更新每个粒子的速度位置:对于第i个粒子,分别计算其个体和全局最优解的影响,然后更新该粒子的速度和位置。
  2. 重新计算每个粒子的适应度
  3. 判断是否满足停止条件,若满足停止条件,则输出当前种群中最优的粒子。

否则,返回第3步进行迭代。

PSO算法的优点是可以适应任意目标函数,复杂度不高,具有较好的搜索能力。根据个体最优和全局最优的影响参数的设置不同会影响算法收敛性和导致算法陷入局部最优。PSO算法由于其具有较强的随机性,导致其搜索过程缺乏自适应能力,可能会在搜索过程中出现较大波动。同时,在高维空间中表现欠佳,并且对于复杂问题,需要精细地调节参数才能取得良好的效果。
image.png

2.PSO更新方法🌾

粒子群算法的核心就在于更新粒子的速度和位置,从而不断搜索搜寻空间寻找最优解。其速度和位置的更新公式包括两个部分:个体最优引导和全局最优引导。
以下是粒子群算法的速度和位置更新公式:

  1. 速度更新公式:

$v_{i,d}(t+1) = wv_{i,d}(t) + c_{1}r_{1}(pbest_{i,d}-x_{i,d}(t)) + c_{2}r_{2}(gbest_{d}-x_{i,d}(t))$
其中,$v_{i,d}(t)$ 表示第 i 个粒子在第 d 维的速度,$t$表示迭代次数,$w$表示惯性权重,$c_1$和$c_2$ 表示粒子个体和群体认知系数,$r_1$ 和 $r_2$ 是 0 到 1 之间的随机数,$pbest_{i,d}$ 表示第 i 个粒子在第 d 维度的个体最优位置,$gbest_{d}$ 表示全局最优位置在第 d 维度的取值,$x_{i,d}(t)$ 是第 i 个粒子在第 d 维度的位置。$pbest$是于该粒子之前最优的解,$gbest$是于所有粒子之前最优的解。
公式中第一项是粒子本身的惯性,第二项是粒子向个体最优点的引力,第三项是粒子向全局最优点的引力。

  1. 位置更新公式:

$x_{i,d}(t+1) = x_{i,d}(t) + v_{i,d}(t+1)$
其中 $x_{i,d}(t)$ 表示第 i 个粒子在维度 d 上的位置,$v_{i,d}(t+1)$ 表示第 i 个粒子在维度 d 上的速度。通过以上公式,粒子群不断迭代,直到达到所要求的精度或者达到最大迭代次数。

在实际应用中,惯性权重 w 是一个非常重要的参数,控制着算法的局部变化和全局探索的平衡。在迭代过程中,w 的值通常逐渐递减。同时,也需要通过多次实验和调整,确定其他参数的合理取值,以获得更好的优化结果。

  
  

image.png

3.PSO求解TSP问题🌴

实现粒子群算法求解TSP问题的代码:

定义一个 Particle 类用于存储粒子的相关信息;
initialize() 方法来初始化种群;
updatePBest() 方法用于更新个体历史最优路径;
updateVelocity() 方法更新粒子速度;
updatePosition() 方法更新粒子位置;
定义了 calculateFitness() 方法用于计算适应度(路径长度)。
main() 方法中,我们首先调用 initialize() 方法初始化种群,然后通过迭代不断更新粒子的速度、位置、适应度等相关参数,更新个体最优路径和全局最优路径
最终输出最优解路径和路径长度。

public class PSOTSP {
    // 城市集合,第i号城市的编码为i
    public static int[][] coordinates = {
            {200,  40}, {180,  80}, {160, 100}, {140, 120}, {160, 140}, 
            {180, 160}, {200, 200}, {220, 160}, {240, 140}, {260,120}, 
            {240, 100}, {220,  60}, {200,  20}, {240,  20}, {260,  40}
    };
    // 城市数目
    public static int cityNum = coordinates.length;
    // 种群规模
    public static int populationSize = 50;
    // 最大迭代次数
    public static int maxGeneration = 200;
    // 学习因子,用于控制粒子速度和位置的更新
    public static double learningFactor = 0.7;
    // 粒子群
    public static Particle[] particles = new Particle[populationSize];
    // 全局最优解
    public static int[] gbest = new int[cityNum];
    // 全局最优解路径长度
    public static double gbestValue = Double.MAX_VALUE;
    // 计算城市间距离,采用欧氏距离
    public static double[][] distanceMatrix = new double[cityNum][cityNum];
    static {
        for (int i = 0; i < cityNum; i++) {
            for (int j = i + 1; j < cityNum; j++) {
                int xDistance = coordinates[i][0] - coordinates[j][0];
                int yDistance = coordinates[i][1] - coordinates[j][1];
                distanceMatrix[i][j] = Math.sqrt(xDistance * xDistance + yDistance * yDistance);
                distanceMatrix[j][i] = distanceMatrix[i][j];
            }
        }
    }
    public static void main(String[] args) {
        initialize();
        for (int g = 0; g < maxGeneration; g++) {
            for (int i = 0; i < populationSize; i++) {
                for (int j = 0; j < cityNum; j++) {
                    particles[i].updateVelocity(gbest, learningFactor);
                    particles[i].updatePosition();
                    particles[i].calculateFitness(distanceMatrix);
                }
                // 更新个体最优解
                particles[i].updatePBest();
                // 更新全局最优解
                if (particles[i].getPBestValue() < gbestValue) {
                    gbestValue = particles[i].getPBestValue();
                    System.arraycopy(particles[i].getPBest(), 0, gbest, 0, cityNum);
                }
            }
        }
        System.out.print("最优解路径: ");
        for (int i = 0; i < cityNum; i++) {
            System.out.print(gbest[i] + 1 + " ");
        }
        System.out.println();
        System.out.println("最短路径长度: " + gbestValue);
    }
    public static void initialize() {
        // 初始化粒子
        for (int i = 0; i < populationSize; i++) {
            particles[i] = new Particle(cityNum);
            particles[i].initialize();
            particles[i].calculateFitness(distanceMatrix);
            // 更新全局最优解
            if (particles[i].getPBestValue() < gbestValue) {
                gbestValue = particles[i].getPBestValue();
                System.arraycopy(particles[i].getPBest(), 0, gbest, 0, cityNum);
            }
        }
    }
    // 定义粒子实体类
    public static class Particle {
        // 存储粒子经过的路径,初始化为-1
        private int[] path;
        // 存储粒子历史最优路径
        private int[] pbest;
        // 粒子当前位置的适应度值
        private double fitness;
        // 粒子历史最优位置的适应度值
        private double pbestValue;
        // 存储粒子当前速度
        private double[] velocity;
        // 最大速度
        private static double vmax = 5;
        public Particle(int length) {
            path = new int[length];
            pbest = new int[length];
            velocity = new double[length];
        }
        // 随机初始化粒子位置和速度
        public void initialize() {
            List<Integer> list = new ArrayList<Integer>();
            for (int i = 0; i < path.length; i++) {
                list.add(i);
                path[i] = -1;
                velocity[i] = 0;
            }
            Collections.shuffle(list);
            for (int i = 0; i < path.length; i++) {
                path[i] = list.get(i);
            }
            System.arraycopy(path, 0, pbest, 0, path.length);
            pbestValue = fitness;
        }
        // 计算当前位置的适应度值
        public void calculateFitness(double[][] distanceMatrix) {
            double pathLength = 0;
            for (int i = 0; i < path.length - 1; i++) {
                pathLength += distanceMatrix[path[i]][path[i + 1]];
            }
            pathLength += distanceMatrix[path[path.length - 1]][path[0]];
            fitness = pathLength;
        }
        // 更新个体历史最优位置
        public void updatePBest() {
            if (fitness < pbestValue) {
                System.arraycopy(path, 0, pbest, 0, path.length);
                pbestValue = fitness;
            }
        }
        // 更新粒子速度
        public void updateVelocity(int[] gbest, double learningFactor) {
            // 计算个体和全局最优差距
            double[] deltaPbest = new double[path.length];
            double[] deltaGbest = new double[path.length];
            for (int i = 0; i < path.length; i++) {
                deltaPbest[i] = pbest[i] - path[i];
                deltaGbest[i] = gbest[i] - path[i];
            }
            for (int i = 0; i < path.length; i++) {
                // 估算粒子的局部最优解和全局最优解的距离占所有点之间距离的比值
                double alpha = Math.random() * learningFactor;
                // 更新速度
                velocity[i] = alpha * velocity[i] + deltaPbest[i] * Math.random()
                        + (1 - alpha) * deltaGbest[i] * Math.random();
                // 控制速度不超过最大限制
                if (velocity[i] > vmax) {
                    velocity[i] = vmax;
                } else if (velocity[i] < -vmax) {
                    velocity[i] = -vmax;
                }
            }
        }
        // 更新粒子位置
        public void updatePosition() {
            List<Integer> list = new ArrayList<Integer>();
            for (int i = 0; i < path.length; i++) {
                list.add(path[i]);
            }
            for (int i = 0; i < path.length; i++) {
                int index = list.indexOf(i);
                int moveLength = (int) Math.round(velocity[i]);
                int newPosition = (index + moveLength + path.length) % path.length;
                if (newPosition < list.size() && newPosition > 0) {
                    list.add(newPosition, i);
                    list.remove(index < 0 ? list.size() - 1 : index);
                }
            }
            for (int i = 0; i < path.length; i++) {
                path[i] = list.get(i);
            }
        }
        public double getPBestValue() {
            return pbestValue;
        }
        public int[] getPBest() {
            return pbest;
        }
    }
}

☕物有本末,事有终始,知所先后。🍭

相关文章
WK
|
2月前
|
机器学习/深度学习 算法 数据挖掘
PSO算法的应用场景有哪些
粒子群优化算法(PSO)因其实现简单、高效灵活,在众多领域广泛应用。其主要场景包括:神经网络训练、工程设计、电力系统经济调度与配电网络重构、数据挖掘中的聚类与分类、控制工程中的参数整定、机器人路径规划、图像处理、生物信息学及物流配送和交通管理等。PSO能处理复杂优化问题,快速找到全局最优解或近似解,展现出强大的应用潜力。
WK
109 1
|
2月前
|
机器学习/深度学习 算法 数据安全/隐私保护
基于PSO粒子群优化的GroupCNN分组卷积网络时间序列预测算法matlab仿真
本项目展示了一种结合粒子群优化(PSO)与分组卷积神经网络(GroupCNN)的时间序列预测算法。该算法通过PSO寻找最优网络结构和超参数,提高预测准确性与效率。软件基于MATLAB 2022a,提供完整代码及详细中文注释,并附带操作步骤视频。分组卷积有效降低了计算成本,而PSO则智能调整网络参数。此方法特别适用于金融市场预测和天气预报等场景。
WK
|
2月前
|
机器学习/深度学习 自然语言处理 算法
PSO算法和人工神经网络有什么不同
PSO算法(粒子群优化)与人工神经网络(ANN)在原理、应用及优化方式上差异显著。PSO模拟鸟群行为,通过粒子协作在解空间中搜索最优解;而ANN模仿大脑神经元结构,通过训练学习输入输出映射,适用于模式识别、图像处理等领域。PSO主要用于优化问题,实时性高,结果直观;ANN则在处理复杂非线性关系方面更强大,但结构复杂,训练耗时长,结果解释性较差。实际应用中需根据需求选择合适技术。
WK
26 0
WK
|
2月前
|
算法 决策智能
PSO算法的缺点有哪些
粒子群优化(PSO)算法是一种基于群体协作的随机搜索方法,源自对鸟群觅食行为的模拟。尽管其在多领域展现了独特优势,但也存在显著缺点:易陷局部最优、搜索精度不足、高度依赖参数设置、理论基础薄弱、适用范围有限及早熟收敛问题。针对这些问题,可通过结合其他优化算法、调整参数及改进更新公式等方式提升其性能。
WK
62 0
|
3月前
|
数据采集 算法
基于PSO粒子群算法的三角形采集堆轨道优化matlab仿真
该程序利用PSO算法优化5个4*20矩阵中的模块采集轨迹,确保采集的物品数量及元素含量符合要求。在MATLAB2022a上运行,通过迭代寻优,选择最佳模块组合并优化轨道,使采集效率、路径长度及时间等综合指标最优。具体算法实现了粒子状态更新、需求量差值评估及轨迹优化等功能,最终输出最优轨迹及其相关性能指标。
|
4月前
|
数据采集 机器学习/深度学习 算法
Python实现用PSO粒子群优化算法对KMeans聚类模型进行优化项目实战
Python实现用PSO粒子群优化算法对KMeans聚类模型进行优化项目实战
108 2
|
4月前
|
机器学习/深度学习 数据采集 算法
Python实现PSO粒子群优化支持向量机回归模型(svr算法)项目实战
Python实现PSO粒子群优化支持向量机回归模型(svr算法)项目实战
255 1
|
5月前
|
算法
m基于PSO粒子群优化的LDPC码NMS译码算法最优归一化参数计算和误码率matlab仿真
MATLAB2022a仿真实现了基于遗传优化的NMS LDPC译码算法,优化归一化参数以提升纠错性能。NMS算法通过迭代处理低密度校验码,而PSO算法用于寻找最佳归一化因子。程序包含粒子群优化的迭代过程,根据误码率评估性能并更新解码参数。最终,展示了迭代次数与优化过程的关系,并绘制了SNR与误码率曲线。
50 2
|
5月前
|
算法
基于PSO粒子群优化的PID控制器参数整定算法matlab仿真
该文探讨了使用PSO(粒子群优化)算法优化PID控制器参数的方法。通过PSO迭代,不断调整PID控制器的Kp、Ki、Kd增益,以减小控制误差。文中提供了MATLAB2022a版本的核心代码,展示了参数优化过程及结果。系统仿真图像显示了参数随迭代优化的变化。PID控制器结合PSO算法能有效提升控制性能,适用于复杂系统的参数整定,未来研究可关注算法效率提升和应对不确定性。
|
5月前
|
算法
m基于PSO粒子群优化的LDPC码OMS译码算法最优偏移参数计算和误码率matlab仿真
MATLAB2022a仿真实现了Offset Min-Sum (OMS)译码算法与粒子群优化(PSO)结合,以优化偏移参数,提升LDPC码解码性能。PSO通过迭代寻找最小化误码率(BER)的最佳偏移量。核心程序运用PSO进行参数更新和适应度函数(BER)评估,最终在不同信噪比下展示OMS解码性能,并保存结果。
72 0