【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现

关键词:粒子群算法、python

一、粒子群算法概述

  粒子群算法,也称粒子群优化算法或鸟群觅食算法(Particle Swarm Optimization),缩写为 PSO,属于进化算法的一种,和模拟退火算法相似,它也是从随机解出发,通过迭代寻找最优解,它也是通过适应度来评价解的品质,但它比遗传算法规则更为简单,它没有遗传算法的“交叉”(Crossover) 和“变异”(Mutation) 操作,它通过追随当前搜索到的最优值来寻找全局最优。这种算法以其实现容易、精度高、收敛快等优点引起了学术界的重视,并且在解决实际问题中展示了其优越性。

  原理讲解可参考:粒子群算法入门及实践、粒子群算法详解,这些例子都有原理,还有java、matlab实现,本例是用Python及相关第三方包实现粒子群算法

二、例题

  我用上面那个博客的例题:f(x, y, m)= \begin{equation} \left\{ \begin{array}{lr} 30x-y; & x<m, y<m& \\ 30y-x; & x<m,y\ge m\\ x^2-\frac{y}{2};&x\ge m,y<m\\ 20y^2-500x;&x\ge m,y\ge m \end{array} \right. \end{equation} \\ 0 \le x \le 60 \quad 0 \le y \le 60 \quad m = 30查找这个函数的最优解

三、粒子群的Python实现

1.首先是引入包

import numpy as np
import pandas as pd

from collections import Iterable, Counter

2.打好框架

  首先我们要明确粒子群算法的步骤,下面是流程图:
【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现_第1张图片
流程图

  我们所要做的步骤,分别是初始化(init)、循环更新位置、速度(Update_position)与最大值记录(Update_best),以及输出结果(info),并且将迭代部分的两步包在一起,形成一个函数(pso)

class PSO:
    
    def __init__(self, func, bound, POP_SIZE, w=1, c1=0.2, c2=0.2, v_max=0.05, *, var_name=None):
        pass
        
    def get_fitness(self):
        pass
    
    def update_position(self):
        pass            
    
    def update_best(self):
        pass
                
    def pso(self):
        pass
        
    def info(self):
        pass

3.__init__分析和实现

  前面已经反复提到过,__init__方法会作为初始化的函数,需要传递方法、参数,这在我们上面定义的函数签名上有很好的体现。
  首先是func参数,按理来说,这里应该是函数(method),不过这是我的习惯,喜欢写方法(function),无伤大雅;这里用来传递函数引用,无论是def出来的函数,还是lambda表达式出来的函数都没问题。
  bound参数为边界,就是变量的取值范围,数据格式为((x_min, x_max), (y_min, y_max), z,...),简单来说就是个容器,里面装着许多个二元容器或单个变量。例如:[(0, 60), (20, 70), (23, 45.5), 7],其中二元容器代表变量的上下限,单个变量为变量只能取得唯一值。此处只支持这些,如果有类似:(min, None)这种单边界或((min1, max1), (min2, max2))这种多边界的需求,可以自己去实现。
  POP_SIZE为粒子数,可以看成一个范围内觅食鸟的个数。
  w、c1、c2是后面更新公式所需要的参数,我们更新的时候再讲解。
  v_max为初始速度的最大值。
  var_name是变量名,None为默认,前面加上*参数的意思是:想要给var_name赋值,必须用var_name=...的形式。
  参数讲解完毕,先上__init__代码:

def __init__(self, func, bound, POP_SIZE, w=1, c1=0.2, c2=0.2, v_max=0.05, *, var_name=None):
    bounds = Counter([isinstance(a, Iterable) for a in bound])[True]
    Var_size = int(np.ceil(POP_SIZE ** (1/bounds)))

    vals = [np.linspace(var[0], var[1], Var_size) if isinstance(var, Iterable) else np.array([var]) for var in bound]
    vals = np.array(list(map(lambda var: var.flatten(), np.meshgrid(*vals))))
    self.var_quantity, self.POP_SIZE = vals.shape
    self.func = func
    self.bound = bound
    self.w = w
    self.c1=c1
    self.c2=c2
    self.v_max = v_max
    self.var_name = var_name

#         粒子
    self.particles = np.array(list(zip(*vals)))

    self.velocity = np.random.rand(*self.particles.shape) * v_max

#         粒子最优位置
    self.person_best = self.particles.copy()
#         全局最优位置
    self.global_best = max(self.person_best, key=lambda particle: self.func(*particle)).copy()

  让我们先看看前五行是什么意思。
  第一行,用计数器数一个推导列表,这个列表由bool类型组成,当前元素含义是:bound参数当前位置元素是否可迭代(既是二元组),然后查里面有多少个True;因此第一行的含义是:看变量中有多少个变量是在范围内取值
  第二行实现的是这个公式:,什么含义呢?想想,假如我们有、两个变量,各取10个数字,那么的元组是否就有100种可能的取值?以此推广,假如有三个变量,各取5个值,元组就有125种可能取值;假如我们需要100个粒子,有三个变量,那么就,结果是5,虽然能得到的是125个粒子,但结果近似;这里的算法不但是根据变量数,还要求变量不能是那种唯一取值的变量,因此开根号数由第一行得到。
  第三行,确定所有变量的取值;如果是范围内取值,将会由其最小值到最大值,个数为上一行计算的进行填充;如果是单个值,就只计入一个值。例如上面的例题,、变量为范围取值,是单个取值,我们需要100个粒子,上面计算的,根据和的取值范围,和的取值列表都为[0, 6.667, 13.333, 20, 26.667, 33.333, 40, 46,667, 53,555, 60],而的取值列表为[30]。这样能做到像网状一样的均匀取值。
  第四行,np.meshgrid函数就能做到找出所有的取值,我们将所有变量传入,它将返回“变量数+1”维的数组,最外面一层可以解包给各个变量,每个变量持有“变量数”维数组,我们可以将其看做“变量数-1”维数组,每一个维度为其他变量的取值个数,最内层的元素是原来的变量本身。听起来很绕口,自己试试就明白了,这个方法常用于多维作图铺开取值网格。不过多维的数组处理起来很头疼,因此我将它们都用fatten铺成了一维数组,原来位置的对应关系没有改变,现在vals成了二维数组,一维数的含义是变量数,二维数含义是最终粒子个数。这就是第五行做的事。
  后面一堆参数赋值没什么好说的。
  然后是粒子注释的下一行,vals变量的操作还是有些恶心,不符合我的处理思维,因此我将其转置(我常用的转置操作,不过vals是numpy.array类型,所以用self.particles=vals.T就能实现),这样就会变成一个容器,内部的每一个元素都是位置的集合。
  velocity是和粒子同样维度的随机序列,意思是每一个粒子的每一个变量都有速度。
  最后记录当前位置为粒子最优位置。

4.get_fitness

  就是获得所有粒子的适应度,因此将粒子容器中的每一个粒子代入到函数func中计算即可。

def get_fitness(self):
    return np.array([self.func(*particle) for particle in self.particles])

  之前我写函数直接return func(*self.particles),这样能调用numpy中数组的矩阵运算,但这个函数中存在判断,因此不能简单送入方法,而选择用列表推导实现。

5.update_position

  更新位置和速度的函数,让我们先看数学的式子:其中:
\begin{equation} \begin{array}{lr} V_{iD}^{k} \qquad& k时刻,第i个粒子的第D个变量的速度\\ \omega & 惯性因子\\ c_1 & 自记录的学习步长 \\ c_2 & 群体记录的学习步长 \\ r_1、r_2 & [0, 1]间的随机数 \\ p_{id}^k & k时刻第i个粒子最大适应度所在的位置的第D个变量取值 \\ p_{gd}^k & k时刻之前所有记录中最大适应度所在的位置的第D个变量取值 \\ x_{id}^k & k时刻第i个粒子的第D个变量的取值 \end{array} \end{equation}  这里,是__init__中随机的,是__init__中均匀铺开的粒子,、、是需要我们试验填入的参数,自身最优和全局全时刻最优在__init__中已被初始化,并在后续会有更新。
  怎么理解上面的公式呢?
  首先是速度,当前速度与四个因素有关:之前的速度、之前的位置、全局最优位置、自身最优位置。
  自身速度会有一定程度保留下来,惯性因子就决定了保留速度的程度。
  后面两个式子,我们想,假如觅食中的鸟知道种群在哪里找到的食物最多,也记住了自己在哪里找到了最多的食物,假如当前位置不尽如人意,就会向这两个地方赶,而离的越远,就要越快地赶去,因此位置两者之差越大,对速度加成越高。加成的程度、则是要我们多次试验,、提供了随机度。
  可以看下面的图:

【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现_第2张图片
灵魂画师时刻

  红底为鸟群得到的食物分布情况,当前的黑色粒子正以浅绿色的方向运动,下一时刻它的运行轨迹,同时被三个力拉扯:原有的行动方向,自身记录最高点,全局记录最高点。是不是有点像我们之前学的受力分解?
  根据公式,我们写出更新位置和速度的代码

def update_position(self):
    for index, particle in enumerate(self.particles):
        V_k_plus_1 = self.w * self.velocity[index] \
                            + self.c1*np.random.rand()*(self.person_best[index]-particle) \
                            + self.c2*np.random.rand()*(self.global_best-particle)

        self.particles[index] =  self.particles[index] + V_k_plus_1
        self.velocity[index] = V_k_plus_1

        for i, var in enumerate(particle):
            if isinstance(self.bound[i], Iterable):
                if var < self.bound[i][0]:
                    self.particles[index][i] = self.bound[i][0]
                elif var > self.bound[i][1]:
                    self.particles[index][i] = self.bound[i][1]
            elif var != self.bound[i]:
                self.particles[index][i] = self.bound[i]

  上面一段先算出的速度,然后改变位置和原有速度;下面一段是用于检查,毕竟变量是有取值范围的,如果超出范围,将重置为边界,如果是唯一取值,将让值不变。

6.update_best

  没什么好说的,就是更新最优记录

def update_best(self):
    global_best_fitness = self.func(*self.global_best)
    person_best_value = np.array([self.func(*particle) for particle in self.person_best])

    for index, particle in enumerate(self.particles):
        current_particle_fitness = self.func(*particle)

        if current_particle_fitness > person_best_value[index]:
            person_best_value[index] = current_particle_fitness
            self.person_best[index] = particle
        if current_particle_fitness > global_best_fitness:
            global_best_fitness = current_particle_fitness
            self.global_best = particle

7.pso

  执行两个更新

def pso(self):
    self.update_position()
    self.update_best()

8.info

  输出当前粒子信息,如果没传入变量名(var_name为None),就默认采用这种形式。

def info(self):
    result = pd.DataFrame(self.particles)
    if self.var_name == None:
        result.columns = [f'x{i}' for i in range(len(self.bound))]
    else:
        result.columns = self.var_name
    result['fitness'] = self.get_fitness()
    return result

四、实验

  我们准备好需要传递的参数,传入构造方法里面

func = lambda x, y, m: 30*x-y if x < m and y < m else 30*y-x if x=m else x**2-y/2 if x>=m and y

  然后不断迭代,由于粒子群算法属于群体进化,因此我们可以打印总体的加和方式,体现进化的效果,并打印出全体最优解及最优取值:

for _ in range(1000):
    pso.pso()
    print(pso.get_fitness().sum())
    
print(pso.global_best, func(*pso.global_best))
【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现_第3张图片

输出结果上面还有,省略不放了。
  此时我们调用info即可看到格式化的结果输出:

pso.info()
【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现_第4张图片

  可以看出,总体进化不是很理想,但有不少个体已经接近全局最优,不断调试参数,或许能改善这种情况。

五、总结

  和遗传算法一样,粒子群算法是对现实中规律的总结和应用,具体怎么使用,要看我们如何抽象问题;此例是解决一个普通的数学优化问题,但假如我们对其他问题抽象,只要能通过一系列参数得到一个值,就可抽象为适应度函数;只要能体现过去状态、最优记录,就能抽象为为pso算法。
  如果我们遇到其他问题,即使和当前问题不是很像,但动脑抽象,大致贴和相关算法,说不定就能得到意外的解决方案。

你可能感兴趣的:(【机器学习】粒子群算法(Particle Swarm Optimization)的Python实现)